<template>
    <transition name="fade-and-fly">
        <div v-if="!isReady" class="fixed top-0 left-0 w-screen h-screen flex items-center justify-center overflow-hidden">
            <!-- Falling stars -->
            <div v-if="$store.state.theme_vars.activate_dropping_animation" class="absolute inset-0 pointer-events-none overflow-hidden">
                <div v-for="n in 50" :key="n" class="absolute animate-fall animate-giggle"
                    :style="{
                    left: `${Math.random() * 100}%`,
                    top: `-20px`,
                    animationDuration: `${3 + Math.random() * 7}s`,
                    animationDelay: `${Math.random() * 5}s`
                    }">
                {{$store.state.theme_vars.falling_object}}
                </div>
            </div>

            <div class="flex flex-col items-center text-center max-w-4xl w-full px-4 relative z-10">
                <div class="mb-8 w-full">
                    <div class="text-5xl md:text-6xl font-bold mb-2 hover:scale-105 transition-transform lollms-title-style">
                        {{$store.state.theme_vars.lollms_title}}
                    </div>


                    <p class="text-2xl italic">
                        One tool to rule them all
                    </p>
                    <p class="text-xl mb-6">
                        by ParisNeo
                    </p>
                    <p class="bottom-0 text-2xl italic">
                        {{ version_info }}
                    </p>

                    <!-- Clickable interesting fact card -->
                    <div class="interesting-facts transition-transform duration-300 cursor-pointer"
                        @click="updateRandomFact">
                        <p class="text-lg ">
                            <span class="font-semibold">🤔 Fun Fact: </span>
                            <span v-html="randomFact"></span>
                        </p>
                    </div>


                    <!-- Animated Progress Bar -->
                    <div class="animated-progressbar-bg">
                        <!-- Progress Background -->
                        <div class="animated-progressbar-fg"
                            :style="{ width: `${loading_progress}%` }">
                        </div>
                        <!-- Star that moves with progress -->
                        <div class="absolute top-0 h-full flex items-center transition-all duration-300"
                            :style="{ left: `${loading_progress}%`, transform: 'translateX(-50%)' }">
                        </div>
                    </div>
                </div>
                
                <div class="w-full max-w-2xl">
                    <div role="status" class="w-full">
                        <p class="text-xl">
                            {{ loading_infos }}...
                        </p>
                        <p class="text-2xl font-bold mt-2">
                            {{ Math.round(loading_progress) }}%
                        </p>
                    </div>
                </div>
            </div>
        </div>
    </transition>
    <transition name="slide-right">
    <div  v-if="showLeftPanel"
        class="relative flex flex-col no-scrollbar shadow-lg w-[16rem] "
        >
            <RouterLink :to="{ name: 'discussions' }" class="flex items-center space-x-2"> <!-- Added space-x-2 -->
                <div class="logo-container"> <!-- Removed mr-1 -->
                <img class="w-12 h-12 rounded-full object-cover logo-image" 
                    :src="$store.state.config == null ? storeLogo : $store.state.config.app_custom_logo != '' ? '/user_infos/' + $store.state.config.app_custom_logo : storeLogo" 
                    alt="Logo" title="LoLLMS WebUI">
                </div>
                <div class="flex flex-col justify-center">
                    <div class="text-center p-2">
                        <div class="text-md relative inline-block">
                            <span class="relative inline-block font-bold tracking-wide text-black dark:text-white">
                                LoLLMS
                            </span>
                            <div class="absolute -bottom-0.5 left-0 w-full h-0.5 
                                        bg-black dark:bg-white
                                        transform origin-left transition-transform duration-300
                                        hover:scale-x-100 scale-x-0"></div>
                        </div>
                    </div>


                <p class="text-gray-400 text-sm">One tool to rule them all</p>
                </div>
            </RouterLink>

            <div class="toolbar discussion">
                <!-- Toolbar container -->
                <div class="toolbar-container">                    
                    <!-- "+" button -->
                    <button 
                        class="toolbar-button" 
                        title="Create new discussion" 
                        @click="createNewDiscussion"
                    >
                        <i data-feather="plus"></i>
                    </button>
                    <div class="toolbar-button" @mouseleave="hideSkillsLibraryMenu" v-if="!loading">
                        <!-- Expandable menu positioned above the button -->
                        <div v-show="isSkillsLibraryMenuVisible" @mouseenter="showSkillsLibraryMenu" class="absolute m-0 p-0 z-50 top-full left-0 transform bg-white dark:bg-bg-dark rounded-lg shadow-lg ring-1 ring-black ring-opacity-5 focus:outline-none transition-all duration-300 ease-out mb-2">
                            <div class="p-4 flex flex-wrap gap-2 items-center">
                                <!-- Add to skills database -->
                                <button 
                                    v-if="!loading" 
                                    type="button" 
                                    @click.stop="addDiscussion2SkillsLibrary" 
                                    title="Add this discussion content to skills database" 
                                    class="text-3xl hover:text-secondary dark:hover:text-secondary-light duration-150 active:scale-95"
                                >
                                    <i data-feather="plus"></i>
                                </button>

                                <!-- Toggle skills database -->
                                <button 
                                    v-if="!loading && $store.state.config.activate_skills_lib" 
                                    type="button" 
                                    @click.stop="toggleSkillsLib" 
                                    title="Skills database is activated" 
                                    class="text-3xl hover:text-secondary dark:hover:text-secondary-light duration-150 active:scale-95"
                                >
                                    <i data-feather="check-circle"></i>
                                </button>
                                <button 
                                    v-if="!loading && !$store.state.config.activate_skills_lib" 
                                    type="button" 
                                    @click.stop="toggleSkillsLib" 
                                    title="Skills database is deactivated" 
                                    class="text-3xl hover:text-secondary dark:hover:text-secondary-light duration-150 active:scale-95"
                                >
                                    <i data-feather="x-octagon"></i>
                                </button>
                                
                                <!-- Show skills database -->
                                <button 
                                    v-if="!loading" 
                                    type="button" 
                                    @click.stop="showSkillsLib" 
                                    title="Show Skills database" 
                                    class="text-3xl hover:text-secondary dark:hover:text-secondary-light duration-150 active:scale-95"
                                >
                                    <i data-feather="book"></i>
                                </button>
                            </div>
                        </div>
                        <!-- Menu toggle button -->
                        <div @mouseenter="showSkillsLibraryMenu" class="menu-hover-area">
                            <button class="w-8 h-8" title="Toggle Skills library menu">
                                <svg width="24" height="24" viewBox="0 0 24 24" xmlns="http://www.w3.org/2000/svg">
                                <path d="M4 4v16h16V4H4zm2 2h12v12H6V6zm2 2h2v8H8V8zm3 0h2v8h-2V8zm3 0h2v8h-2V8z" fill="currentColor"/>
                                </svg>

                            </button>
                        </div>                        
                    </div>
                    <div class="toolbar-button" @mouseleave="hideMenu" v-if="!loading">
                        <!-- Expandable menu positioned above the button -->
                        <div v-show="isMenuVisible" @mouseenter="showMenu" class="absolute m-0 p-0 z-50 top-full left-0 transform bg-white dark:bg-bg-dark rounded-lg shadow-lg ring-1 ring-black ring-opacity-5 focus:outline-none transition-all duration-300 ease-out mb-2">
                            <div class="p-4 flex flex-wrap gap-2 items-center">
                                <!-- Edit discussion list -->
                                <button 
                                    class="text-3xl hover:text-secondary dark:hover:text-secondary-light duration-150 active:scale-95" 
                                    title="Edit discussion list" 
                                    type="button" 
                                    @click="isCheckbox = !isCheckbox" 
                                    :class="isCheckbox ? 'text-secondary dark:text-secondary-light' : 'text-gray-700 dark:text-gray-300'"
                                >
                                    <i data-feather="check-square"></i>
                                </button>

                                <!-- Reset database -->
                                <button 
                                    class="text-3xl hover:text-red-500 dark:hover:text-red-400 duration-150 active:scale-95" 
                                    title="Reset database, remove all discussions" 
                                    @click.stop=""
                                >
                                    <i data-feather="trash-2"></i>
                                </button>

                                <!-- Export database -->
                                <button 
                                    class="text-3xl hover:text-secondary dark:hover:text-secondary-light duration-150 active:scale-95" 
                                    title="Export database" 
                                    type="button" 
                                    @click.stop="database_selectorDialogVisible=true"
                                >
                                    <i data-feather="database"></i>
                                </button>

                                <!-- Import discussions -->
                                <div class="relative">
                                    <input type="file" ref="fileDialog" class="hidden" @change="importDiscussions" />
                                    <button 
                                        class="text-3xl hover:text-secondary dark:hover:text-secondary-light duration-150 active:scale-95 rotate-90" 
                                        title="Import discussions" 
                                        type="button" 
                                        @click.stop="$refs.fileDialog.click()"
                                    >
                                        <i data-feather="log-in"></i>
                                    </button>
                                </div>

                                <!-- Import discussion bundle -->
                                <div class="relative">
                                    <input type="file" ref="bundleLoadingDialog" class="hidden" @change="importDiscussionsBundle" />
                                    <button 
                                        v-if="!showSaveConfirmation" 
                                        title="Import discussion bundle" 
                                        @click.stop="$refs.bundleLoadingDialog.click()" 
                                        class="text-3xl hover:text-secondary dark:hover:text-secondary-light duration-150 active:scale-95"
                                    >
                                        <i data-feather="folder"></i>
                                    </button>
                                </div>


                                <!-- Loading spinner -->
                                <div v-if="loading" title="Loading.." class="flex justify-center">
                                    <div role="status">
                                        <svg aria-hidden="true" class="w-8 h-8 animate-spin fill-secondary dark:fill-secondary-light" viewBox="0 0 100 101" fill="none" xmlns="http://www.w3.org/2000/svg">
                                            <path d="M100 50.5908C100 78.2051 77.6142 100.591 50 100.591C22.3858 100.591 0 78.2051 0 50.5908C0 22.9766 22.3858 0.59082 50 0.59082C77.6142 0.59082 100 22.9766 100 50.5908ZM9.08144 50.5908C9.08144 73.1895 27.4013 91.5094 50 91.5094C72.5987 91.5094 90.9186 73.1895 90.9186 50.5908C90.9186 27.9921 72.5987 9.67226 50 9.67226C27.4013 9.67226 9.08144 27.9921 9.08144 50.5908Z" fill="currentColor" />
                                            <path d="M93.9676 39.0409C96.393 38.4038 97.8624 35.9116 97.0079 33.5539C95.2932 28.8227 92.871 24.3692 89.8167 20.348C85.8452 15.1192 80.8826 10.7238 75.2124 7.41289C69.5422 4.10194 63.2754 1.94025 56.7698 1.05124C51.7666 0.367541 46.6976 0.446843 41.7345 1.27873C39.2613 1.69328 37.813 4.19778 38.4501 6.62326C39.0873 9.04874 41.5694 10.4717 44.0505 10.1071C47.8511 9.54855 51.7191 9.52689 55.5402 10.0491C60.8642 10.7766 65.9928 12.5457 70.6331 15.2552C75.2735 17.9648 79.3347 21.5619 82.5849 25.841C84.9175 28.9121 86.7997 32.2913 88.1811 35.8758C89.083 38.2158 91.5421 39.6781 93.9676 39.0409Z" fill="currentFill" />
                                        </svg>
                                        <span class="sr-only">Loading...</span>
                                    </div>
                                </div>

                                <!-- Save confirmation -->
                                <div v-if="showSaveConfirmation" class="flex justify-center space-x-4">
                                    <button class="text-3xl hover:text-red-500 dark:hover:text-red-400 duration-150 active:scale-95" title="Cancel" type="button" @click.stop="showSaveConfirmation = false">
                                        <i data-feather="x"></i>
                                    </button>
                                    <button class="text-3xl hover:text-green-500 dark:hover:text-green-400 duration-150 active:scale-95" title="Confirm save changes" type="button" @click.stop="save_configuration()">
                                        <i data-feather="check"></i>
                                    </button>
                                </div>

                                <!-- Import options -->
                                <div v-if="isOpen" class="flex flex-col space-y-2">
                                    <button @click="importDiscussions" class="text-sm hover:text-secondary dark:hover:text-secondary-light">LOLLMS</button> 
                                    <button @click="importChatGPT" class="text-sm hover:text-secondary dark:hover:text-secondary-light">ChatGPT</button>
                                </div>
                            </div>
                        </div>

                        <!-- Menu toggle button -->
                        <div @mouseenter="showMenu" class="menu-hover-area">
                            <button class="w-8 h-8" title="Toggle menu">
                                <i data-feather="menu"></i>
                            </button>
                        </div>
                    </div>




                    <div class="toolbar-button" @mouseleave="hideBindingsMenu" v-if="!loading">
                        <div class="relative inline-block">
                            <!-- Bindings menu positioned above the button -->
                            <div v-show="isBindingsMenuVisible" @mouseenter="showBindingsMenu" class="absolute m-0 p-0 z-10 top-full left-0 transform w-80 bg-white dark:bg-gray-900 rounded-md shadow-lg ring-1 ring-black ring-opacity-5 focus:outline-none transition-all duration-300 ease-out mb-2">
                                <!-- Search Bar -->
                                <div class="p-2 border-b border-gray-200 dark:border-gray-700">
                                    <input 
                                        type="text" 
                                        v-model="bindingSearchQuery" 
                                        placeholder="Search bindings..." 
                                        class="w-full px-3 py-2 rounded-md border border-gray-300 dark:border-gray-600 dark:bg-gray-800 dark:text-white focus:outline-none focus:ring-2 focus:ring-blue-500"
                                    >
                                </div>
                                
                                <div class="p-4 grid grid-cols-3 gap-4 max-h-80 overflow-y-auto custom-scrollbar">
                                    <div 
                                        v-for="(item, index) in filteredBindings" 
                                        :key="index" 
                                        class="relative group/item flex flex-col items-center"
                                    >                             
                                        <!-- Icon and Name Container -->
                                        <div class="flex flex-col items-center hover:bg-blue-100 dark:hover:bg-blue-900 p-2 rounded-md w-full cursor-pointer">
                                            <button 
                                                @click.prevent="setBinding(item)" 
                                                :title="item.name" 
                                                class="w-12 h-12 rounded-md overflow-hidden transition-transform duration-200 transform group-hover/item:scale-105 focus:outline-none"
                                            >
                                                <img 
                                                    :src="item.icon ? item.icon : modelImgPlaceholder" 
                                                    @error="modelImgPlaceholder" 
                                                    :alt="item.name" 
                                                    class="w-full h-full object-cover" 
                                                    :class="{'border-2 border-secondary': item.name == binding_name}"
                                                >
                                            </button>
                                            
                                            <!-- Truncated Name -->
                                            <span class="mt-1 text-xs text-center w-full truncate" :title="item.name">
                                                {{ item.name }}
                                            </span>
                                        </div>

                                        <!-- Hover Options -->
                                        <div 
                                            class="absolute top-0 left-0 w-full h-full opacity-0 group-hover/item:opacity-100 transition-opacity duration-200 bg-white dark:bg-gray-900 rounded-md shadow-md p-2 flex flex-col items-center justify-center"
                                        >
                                            <!-- Full Name -->
                                            <span class="text-xs font-medium mb-2 text-center"
                                            @click.prevent="setBinding(item)"                                             
                                            >{{ item.name }}</span>
                                            
                                            <!-- Action Buttons -->
                                            <div class="flex space-x-1">
                                                <button 
                                                    @click.prevent="showModelConfig(item)" 
                                                    class="p-1 bg-blue-500 rounded-full text-white hover:bg-blue-600 focus:outline-none" 
                                                    title="Configure Binding"
                                                >
                                                    <svg class="w-3 h-3" fill="none" stroke="currentColor" viewBox="0 0 24 24" xmlns="http://www.w3.org/2000/svg">
                                                        <path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M10.325 4.317c.426-1.756 2.924-1.756 3.35 0a1.724 1.724 0 002.573 1.066c1.543-.94 3.31.826 2.37 2.37a1.724 1.724 0 001.065 2.572c1.756.426 1.756 2.924 0 3.35a1.724 1.724 0 00-1.066 2.573c.94 1.543-.826 3.31-2.37 2.37a1.724 1.724 0 00-2.572 1.065c-.426 1.756-2.924 1.756-3.35 0a1.724 1.724 0 00-2.573-1.066c-1.543.94-3.31-.826-2.37-2.37a1.724 1.724 0 00-1.065-2.572c-1.756-.426-1.756-2.924 0-3.35a1.724 1.724 0 001.066-2.573c-.94-1.543.826-3.31 2.37-2.37.996.608 2.296.07 2.572-1.065z"></path>
                                                        <path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M15 12a3 3 0 11-6 0 3 3 0 016 0z"></path>
                                                    </svg>
                                                </button>
                                            </div>
                                        </div>
                                    </div>
                                </div>
                            </div>

                            <div @mouseenter="showBindingsMenu" class="bindings-hover-area">
                                <button @click.prevent="showModelConfig()" class="w-6 h-6">
                                    <img :src="currentBindingIcon"
                                        class="w-6 h-6 rounded-full object-fill text-red-700 border-2 active:scale-90 hover:border-secondary hover:scale-110 hover:-translate-y-1 duration-200"
                                        :title="currentBinding ? currentBinding.name : 'unknown'">
                                </button>
                            </div>
                        </div>
                    </div>

                    <div class="toolbar-button" @mouseleave="hideModelsMenu" v-if="!loading">
                        <div class="relative inline-block">
                            <!-- Models menu positioned above the button -->
                            <div v-show="isModelsMenuVisible" @mouseenter="showModelsMenu" class="absolute m-0 p-0 z-10 top-full left-0 transform w-80 bg-white dark:bg-gray-900 rounded-md shadow-lg ring-1 ring-black ring-opacity-5 focus:outline-none transition-all duration-300 ease-out mb-2">
                                <!-- Search Bar -->
                                <div class="p-2 border-b border-gray-200 dark:border-gray-700">
                                    <input 
                                        type="text" 
                                        v-model="modelSearchQuery" 
                                        placeholder="Search models..." 
                                        class="w-full px-3 py-2 rounded-md border border-gray-300 dark:border-gray-600 dark:bg-gray-800 dark:text-white focus:outline-none focus:ring-2 focus:ring-blue-500"
                                    >
                                </div>
                                
                                <div class="p-4 grid grid-cols-3 gap-4 max-h-80 overflow-y-auto custom-scrollbar">
                                    <div 
                                        v-for="(item, index) in filteredModels" 
                                        :key="index" 
                                        class="relative group/item flex flex-col items-center"
                                    >                             
                                        <!-- Icon and Name Container -->
                                        <div class="flex flex-col items-center hover:bg-blue-100 dark:hover:bg-blue-900 p-2 rounded-md w-full cursor-pointer">
                                            <button 
                                                @click.prevent="setModel(item)" 
                                                :title="item.name" 
                                                class="w-12 h-12 rounded-md overflow-hidden transition-transform duration-200 transform group-hover/item:scale-105 focus:outline-none"
                                            >
                                                <img 
                                                    :src="item.icon ? item.icon : modelImgPlaceholder" 
                                                    @error="personalityImgPlacehodler" 
                                                    :alt="item.name" 
                                                    class="w-full h-full object-cover" 
                                                    :class="{'border-2 border-secondary': item.name == model_name}"
                                                >
                                            </button>
                                            
                                            <!-- Truncated Name -->
                                            <span class="mt-1 text-xs text-center w-full truncate" :title="item.name">
                                                {{ item.name }}
                                            </span>
                                        </div>

                                        <!-- Hover Options -->
                                        <div 
                                            class="absolute top-0 left-0 w-full h-full opacity-0 group-hover/item:opacity-100 transition-opacity duration-200 bg-white dark:bg-gray-900 rounded-md shadow-md p-2 flex flex-col items-center justify-center"
                                        >
                                            <!-- Full Name -->
                                            <span class="text-xs font-medium mb-2 text-center"
                                            @click.prevent="setModel(item)"                                             
                                            >{{ item.name }}</span>
                                            
                                            <!-- Action Buttons -->
                                            <div class="flex space-x-1">
                                                <button 
                                                    @click.prevent="copyModelNameFrom(item.name)" 
                                                    class="p-1 bg-blue-500 rounded-full text-white hover:bg-blue-600 focus:outline-none" 
                                                    title="Copy Model Name"
                                                >
                                                    <svg class="w-3 h-3" fill="none" stroke="currentColor" viewBox="0 0 24 24" xmlns="http://www.w3.org/2000/svg"><path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M8 16H6a2 2 0 01-2-2V6a2 2 0 012-2h8a2 2 0 012 2v2m-6 12h8a2 2 0 002-2v-8a2 2 0 00-2-2h-8a2 2 0 00-2 2v8a2 2 0 002 2z"></path></svg>
                                                </button>
                                            </div>
                                        </div>
                                    </div>
                                </div>
                            </div>

                            <div @mouseenter="showModelsMenu" class="models-hover-area">
                                <button @click.prevent="copyModelName()" class="w-6 h-6">
                                    <img :src="currentModelIcon"
                                        class="w-6 h-6 rounded-full object-fill text-red-700 border-2 active:scale-90 hover:border-secondary hover:scale-110 hover:-translate-y-1 duration-400"
                                        :title="currentModel ? currentModel.name : 'unknown'">
                                </button>
                            </div>
                        </div>
                    </div>

                    <!-- Personalities menu positioned below the dock -->
                    <div class="toolbar-button" @mouseleave="hidePersonalitiesMenu" v-if="!loading">
                        <div class="relative inline-block">
                            <!-- Personalities menu positioned above the button -->
                            <div v-show="isPersonalitiesMenuVisible" @mouseenter="showPersonalitiesMenu" class="absolute m-0 p-0 z-10 top-full left-0 transform w-80 bg-white dark:bg-gray-900 rounded-md shadow-lg ring-1 ring-black ring-opacity-5 focus:outline-none transition-all duration-300 ease-out mb-2">
                                <!-- Search Bar -->
                                <div class="p-2 border-b border-gray-200 dark:border-gray-700">
                                    <input 
                                        type="text" 
                                        v-model="personalitySearchQuery" 
                                        placeholder="Search personalities..." 
                                        class="w-full px-3 py-2 rounded-md border border-gray-300 dark:border-gray-600 dark:bg-gray-800 dark:text-white focus:outline-none focus:ring-2 focus:ring-blue-500"
                                    >
                                </div>
                                
                                <div class="p-4 grid grid-cols-3 gap-4 max-h-80 overflow-y-auto custom-scrollbar">
                                    <div 
                                        v-for="(item, index) in filteredPersonalities" 
                                        :key="index" 
                                        class="relative group/item flex flex-col items-center"
                                    >                             
                                        <!-- Icon and Name Container -->
                                        <div class="flex flex-col items-center hover:bg-blue-100 dark:hover:bg-blue-900 p-2 rounded-md w-full cursor-pointer">
                                            <button 
                                                @click.prevent="onPersonalitySelected(item)" 
                                                :title="item.name" 
                                                class="w-12 h-12 rounded-md overflow-hidden transition-transform duration-200 transform group-hover/item:scale-105 focus:outline-none"
                                            >
                                                <img 
                                                    :src="bUrl + item.avatar" 
                                                    @error="personalityImgPlacehodler" 
                                                    :alt="item.name" 
                                                    class="w-full h-full object-cover" 
                                                    :class="{'border-2 border-secondary': $store.state.active_personality_id == $store.state.personalities.indexOf(item.full_path)}"
                                                >
                                            </button>
                                            
                                            <!-- Truncated Name -->
                                            <span class="mt-1 text-xs text-center w-full truncate" :title="item.name">
                                                {{ item.name }}
                                            </span>
                                        </div>

                                        <!-- Hover Options -->
                                        <div 
                                            class="absolute top-0 left-0 w-full h-full opacity-0 group-hover/item:opacity-100 transition-opacity duration-200 bg-white dark:bg-gray-900 rounded-md shadow-md p-2 flex flex-col items-center justify-center"
                                        >
                                            <!-- Full Name -->
                                            <span class="text-xs font-medium mb-2 text-center" 
                                            @click.prevent="onPersonalitySelected(item)"                                             
                                            >{{ item.name }}</span>
                                            
                                            <!-- Action Buttons -->
                                            <div class="flex space-x-1">
                                                <button 
                                                    @click.prevent="unmountPersonality(item)" 
                                                    class="p-1 bg-red-500 rounded-full text-white hover:bg-red-600 focus:outline-none" 
                                                    title="Unmount"
                                                >
                                                    <svg class="w-3 h-3" fill="none" stroke="currentColor" viewBox="0 0 24 24" xmlns="http://www.w3.org/2000/svg"><path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M6 18L18 6M6 6l12 12"></path></svg>
                                                </button>
                                                <button 
                                                    @click.prevent="remount_personality(item)" 
                                                    class="p-1 bg-blue-500 rounded-full text-white hover:bg-blue-600 focus:outline-none" 
                                                    title="Remount"
                                                >
                                                    <svg class="w-3 h-3" fill="none" stroke="currentColor" viewBox="0 0 24 24" xmlns="http://www.w3.org/2000/svg"><path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M4 4v5h.582m15.356 2A8.001 8.001 0 004.582 9m0 0H9m11 11v-5h-.581m0 0a8.003 8.003 0 01-15.357-2m15.357 2H15"></path></svg>
                                                </button>
                                                <button 
                                                    @click.prevent="handleOnTalk(item)" 
                                                    class="p-1 bg-green-500 rounded-full text-white hover:bg-green-600 focus:outline-none" 
                                                    title="Talk"
                                                >
                                                    <svg class="w-3 h-3" fill="none" stroke="currentColor" viewBox="0 0 24 24" xmlns="http://www.w3.org/2000/svg"><path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M8 10h.01M12 10h.01M16 10h.01M9 16H5a2 2 0 01-2-2V6a2 2 0 012-2h14a2 2 0 012 2v8a2 2 0 01-2 2h-5l-5 5v-5z"></path></svg>
                                                </button>
                                            </div>
                                        </div>
                                    </div>
                                </div>
                            </div>

                            <div class="personalities-container">
                                <div @mouseenter="showPersonalitiesMenu" class="personalities-hover-area">
                                <MountedPersonalities ref="mountedPers" :onShowPersList="onShowPersList" :onReady="onPersonalitiesReady"/>
                                </div>
                            </div>
                        </div>
                    </div>
 
                    
            </div>
        </div>
        <!-- Search bar -->
        <div class="w-auto max-w-md mx-auto p-2">
            <form @submit.prevent="handleSearch" class="relative">
                <!-- Search input container -->
                <div class="flex items-center">
                    <div class="relative flex-grow">
                        <!-- Search input -->
                        <input 
                            type="search" 
                            id="default-search" 
                            class="block w-full h-8 px-8 text-sm border border-gray-300 rounded-md
                                bg-bg-light focus:ring-1 focus:ring-secondary focus:border-secondary 
                                dark:bg-bg-dark dark:border-gray-600 dark:placeholder-gray-400 
                                dark:focus:ring-secondary dark:focus:border-secondary
                                transition-all duration-200"
                            placeholder="Search discussions..." 
                            title="Filter discussions by title" 
                            v-model="filterTitle"
                            @keyup.enter="handleSearch"
                        />

                        <!-- Search icon -->
                        <div class="absolute left-2 top-1/2 -translate-y-1/2">
                            <i data-feather="search" class="w-4 h-4 text-gray-400"></i>
                        </div>

                        <!-- Submit button -->
                        <button 
                            type="submit"
                            class="absolute right-2 top-1/2 -translate-y-1/2
                                text-gray-600 hover:text-secondary 
                                rounded-full hover:bg-gray-100 dark:hover:bg-gray-700
                                focus:ring-1 focus:ring-secondary
                                transition-all duration-150 active:scale-98"
                            title="Search"
                        >
                            <i data-feather="arrow-right" class="w-4 h-4"></i>
                        </button>
                    </div>
                </div>
            </form>
        </div>




        <!-- Checkbox operations -->
        <div v-if="isCheckbox" class="w-full p-4 bg-bg-light dark:bg-bg-dark">
            <div class="flex flex-col space-y-2">
                <p v-if="selectedDiscussions.length > 0">Selected: {{ selectedDiscussions.length }}</p>
                <div v-if="selectedDiscussions.length > 0" class="flex space-x-2">
                    <button v-if="!showConfirmation" class="text-2xl hover:text-red-600 duration-75 active:scale-90" title="Remove selected" type="button" @click.stop="showConfirmation = true">
                    <i data-feather="trash"></i>
                    </button>
                    <div v-if="showConfirmation" class="flex space-x-2">
                    <button class="text-2xl hover:text-secondary duration-75 active:scale-90" title="Confirm removal" type="button" @click.stop="deleteDiscussionMulti">
                        <i data-feather="check"></i>
                    </button>
                    <button class="text-2xl hover:text-red-600 duration-75 active:scale-90" title="Cancel removal" type="button" @click.stop="showConfirmation = false">
                        <i data-feather="x"></i>
                    </button>
                    </div>
                </div>
                <div class="flex space-x-2">
                    <button class="text-2xl hover:text-secondary duration-75 active:scale-90 rotate-90" title="Export selected to a json file" type="button" @click.stop="exportDiscussionsAsJson">
                    <i data-feather="codepen"></i>
                    </button>
                    <button class="text-2xl hover:text-secondary duration-75 active:scale-90 rotate-90" title="Export selected to a markdown file" type="button" @click.stop="exportDiscussions">
                    <i data-feather="folder"></i>
                    </button>
                    <button class="text-2xl hover:text-secondary duration-75 active:scale-90 rotate-90" title="Export selected to a markdown file" type="button" @click.stop="exportDiscussionsAsMarkdown">
                    <i data-feather="bookmark"></i>
                    </button>
                    <button class="text-2xl hover:text-secondary duration-75 active:scale-90" title="Select All" type="button" @click.stop="selectAllDiscussions">
                    <i data-feather="list"></i>
                    </button>
                </div>
            </div>
        </div>
        <!-- LEFT SIDE PANEL -->
        <div id="leftPanel" class="flex flex-col flex-grow overflow-y-scroll overflow-x-hidden custom-scrollbar "
            @dragover.stop.prevent="setDropZoneDiscussion()">

            <div class="relative flex flex-row flex-grow mb-10 z-0  w-full">
                <!-- DISCUSSION LIST -->
                <div class="mx-0 flex flex-col flex-grow  w-full " :class="isDragOverDiscussion ? 'pointer-events-none' : ''">
                    <div id="dis-list" :class="filterInProgress ? 'opacity-20 pointer-events-none' : ''"
                        class="flex flex-col flex-grow w-full pb-80">
                        <TransitionGroup v-if="discussionsList.length > 0" name="discussionsList">
                            <Discussion v-for="(item, index) in discussionsList" :key="item.id" :id="item.id" :title="item.title"
                                :selected="currentDiscussion.id == item.id" :loading="item.loading" :isCheckbox="isCheckbox"
                                :checkBoxValue="item.checkBoxValue"
                                :openfolder_enabled="true"
                                @select="selectDiscussion(item)"
                                @delete="deleteDiscussion(item.id)" 
                                @openFolder="openFolder"
                                @editTitle="editTitle" 
                                @makeTitle="makeTitle"
                                @checked="checkUncheckDiscussion" />
                        </TransitionGroup>
                        <div v-if="discussionsList.length < 1"
                            class="gap-2 py-2 my-2 hover:shadow-md hover:bg-primary-light dark:hover:bg-primary rounded-md p-2 duration-75 group cursor-pointer">
                            <p class="px-3">No discussions are found</p>
                        </div>

                    </div>
                </div>
            </div>
        </div>
        <div class="flex flex-row items-center justify-center panels-color">
            <div class="chat-bar text-center flex items-center" @click="showDatabaseSelector">
                <ChatBarButton>
                <svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="mr-2">
                    <ellipse cx="12" cy="5" rx="9" ry="3"></ellipse>
                    <path d="M21 12c0 1.66-4 3-9 3s-9-1.34-9-3"></path>
                    <path d="M3 5v14c0 1.66 4 3 9 3s9-1.34 9-3V5"></path>
                </svg>
                </ChatBarButton>
                <p class="text-center font-large font-bold text-l drop-shadow-md align-middle">{{ formatted_database_name.replace("_"," ") }}</p>
            </div>
        </div>

    </div>
    </transition>
        <div v-if="isReady" class="relative flex flex-col flex-grow" >
            <div id="messages-list"
                class="w-full z-0 flex flex-col  flex-grow  overflow-y-auto scrollbar"
                :class="isDragOverChat ? 'pointer-events-none' : ''">

                <!-- CHAT AREA -->
                <div class="container pt-4 pb-50 mb-50 w-full">
                    <TransitionGroup v-if="discussionArr.length > 0" name="list">
                        <Message v-for="(msg, index) in discussionArr" 
                            :key="msg.id" :message="msg"  :id="'msg-' + msg.id" :ref="'msg-' + msg.id"
                            :host="host"
                            
                            @copy="copyToClipBoard" @delete="deleteMessage" @rankUp="rankUpMessage"
                            @rankDown="rankDownMessage" @updateMessage="updateMessage" @resendMessage="resendMessage" @continueMessage="continueMessage"
                            :avatar="getAvatar(msg.sender)" />
                        
                        <!-- REMOVED FOR NOW, NEED MORE TESTING -->
                        <!-- @click="scrollToElementInContainer($event.target, 'messages-list')"  -->
                        <div v-if="discussionArr.length < 2 && personality.prompts_list.length > 0" class="w-full rounded-lg m-2 shadow-lg hover:border-primary dark:hover:border-primary hover:border-solid hover:border-2 border-2 border-transparent even:bg-bg-light-discussion-odd dark:even:bg-bg-dark-discussion-odd flex flex-col overflow-hidden p-4 pb-2">
                            <h2 class="text-2xl font-bold mb-6 text-gray-800 dark:text-gray-200">Prompt Examples</h2>
                            <div class="overflow-x-auto flex-grow scrollbar-thin scrollbar-thumb-gray-400 dark:scrollbar-thumb-gray-600 scrollbar-track-gray-200 dark:scrollbar-track-gray-800 scrollbar-thumb-rounded-full scrollbar-track-rounded-full">
                                <div class="flex flex-nowrap gap-6 p-4 min-w-full">
                                    <div 
                                        v-for="(prompt, index) in personality.prompts_list" 
                                        :title="extractTitle(prompt)"
                                        :key="index" 
                                        @click="handlePromptSelection(prompt)"
                                        class="flex-shrink-0 w-[300px] bg-white dark:bg-gray-800 border border-gray-300 dark:border-gray-600 rounded-lg p-6 cursor-pointer hover:shadow-xl transition-all duration-300 ease-in-out transform hover:scale-105 flex flex-col justify-between min-h-[220px] group"
                                    >
                                        <div class="space-y-3">
                                            <h3 
                                                class="font-bold text-lg text-gray-900 dark:text-gray-100 mb-2 truncate"
                                                :title="extractTitle(prompt)"
                                            >
                                                {{ extractTitle(prompt) }}
                                            </h3>
                                            <div 
                                                :title="prompt" 
                                                class="text-base text-gray-700 dark:text-gray-300 overflow-hidden line-clamp-4"
                                            >
                                                {{ getPromptContent(prompt) }}
                                            </div>
                                        </div>
                                        <div class="mt-4 text-sm font-medium text-blue-600 dark:text-blue-400 opacity-0 group-hover:opacity-100 transition-opacity duration-300">
                                            Click to select
                                        </div>
                                    </div>
                                </div>
                            </div>


                            <!-- Enhanced Modal for placeholder inputs with live preview -->
                            <div v-if="showPlaceholderModal" class="fixed inset-0 bg-black bg-opacity-50 flex items-center justify-center z-50">
                                <div class="bg-white dark:bg-gray-800 p-6 rounded-lg shadow-xl max-w-4xl w-full max-h-[90vh] flex flex-col">
                                    <h3 class="text-lg font-semibold mb-4">Fill in the placeholders</h3>
                                    
                                    <!-- Container with flex layout -->
                                    <div class="flex-1 flex flex-col min-h-0">
                                        <!-- Live Preview Section -->
                                        <div class="mb-4 p-4 bg-gray-100 dark:bg-gray-700 rounded-lg">
                                            <h4 class="text-sm font-medium mb-2 text-gray-600 dark:text-gray-400">Live Preview:</h4>
                                            <div class="flex-1 h-[200px] overflow-y-auto scrollbar scrollbar-thumb-gray-400 dark:scrollbar-thumb-gray-500 scrollbar-track-gray-200 dark:scrollbar-track-gray-700 scrollbar-thin rounded-md">
                                                <span class="text-base whitespace-pre-wrap">{{ getPromptContent(previewPrompt) }}</span>
                                            </div>
                                        </div>

                                        <!-- Scrollable Container for Placeholders -->
                                        <div class="flex-1 overflow-y-auto">
                                            <div class="space-y-4">
                                                <div v-for="(placeholder, index) in parsedPlaceholders" :key="placeholder.fullText" class="flex flex-col">
                                                    <label :for="'placeholder-'+index" class="text-sm font-medium mb-1">
                                                        {{ placeholder.label }}
                                                    </label>

                                                    <!-- Single line text input -->
                                                    <input 
                                                        v-if="placeholder.type === 'text'"
                                                        :id="'placeholder-'+index"
                                                        v-model="placeholderValues[index]"
                                                        type="text"
                                                        class="border rounded-md p-2 dark:bg-gray-700 dark:border-gray-600"
                                                        :placeholder="placeholder.label"
                                                        @input="updatePreview"
                                                    >

                                                    <!-- Number input (int) -->
                                                    <input 
                                                        v-if="placeholder.type === 'int'"
                                                        :id="'placeholder-'+index"
                                                        v-model.number="placeholderValues[index]"
                                                        type="number"
                                                        step="1"
                                                        class="border rounded-md p-2 dark:bg-gray-700 dark:border-gray-600"
                                                        @input="updatePreview"
                                                    >

                                                    <!-- Number input (float) -->
                                                    <input 
                                                        v-if="placeholder.type === 'float'"
                                                        :id="'placeholder-'+index"
                                                        v-model.number="placeholderValues[index]"
                                                        type="number"
                                                        step="0.01"
                                                        class="border rounded-md p-2 dark:bg-gray-700 dark:border-gray-600"
                                                        @input="updatePreview"
                                                    >

                                                    <!-- Multiline text input -->
                                                    <textarea 
                                                        v-if="placeholder.type === 'multiline'"
                                                        :id="'placeholder-'+index"
                                                        v-model="placeholderValues[index]"
                                                        rows="4"
                                                        class="border rounded-md p-2 dark:bg-gray-700 dark:border-gray-600"
                                                        @input="updatePreview"
                                                    ></textarea>

                                                    <!-- Code editor -->
                                                    <div v-if="placeholder.type === 'code'" class="border rounded-md overflow-hidden">
                                                        <div class="bg-gray-200 dark:bg-gray-900 p-2 text-sm">
                                                            {{ placeholder.language || 'Plain text' }}
                                                        </div>
                                                        <textarea 
                                                            :id="'placeholder-'+index"
                                                            v-model="placeholderValues[index]"
                                                            rows="8"
                                                            class="w-full p-2 font-mono bg-gray-100 dark:bg-gray-900 border-t"
                                                            @input="updatePreview"
                                                        ></textarea>
                                                    </div>

                                                    <!-- Options (dropdown) -->
                                                    <select 
                                                        v-if="placeholder.type === 'options'"
                                                        :id="'placeholder-'+index"
                                                        v-model="placeholderValues[index]"
                                                        class="border rounded-md p-2 dark:bg-gray-700 dark:border-gray-600"
                                                        @change="updatePreview"
                                                    >
                                                        <option value="" disabled>Select an option</option>
                                                        <option 
                                                            v-for="option in placeholder.options" 
                                                            :key="option" 
                                                            :value="option"
                                                        >
                                                            {{ option }}
                                                        </option>
                                                    </select>
                                                </div>
                                            </div>
                                        </div>
                                    </div>

                                    <!-- Footer with buttons -->
                                    <div class="mt-6 flex justify-end space-x-4">
                                        <button 
                                            @click="cancelPlaceholders"
                                            class="px-4 py-2 text-gray-600 hover:text-gray-800 dark:text-gray-400 dark:hover:text-gray-200"
                                        >
                                            Cancel
                                        </button>
                                        <button 
                                            @click="applyPlaceholders"
                                            class="px-4 py-2 bg-blue-500 text-white rounded-md hover:bg-blue-600"
                                        >
                                            Apply
                                        </button>
                                    </div>
                                </div>
                            </div>

                        </div>


                    </TransitionGroup>
                    <WelcomeComponent v-if="!currentDiscussion.id" />
                    <div><br><br><br><br><br><br><br></div>
                </div>

                <div
                    class="absolute w-full bottom-0 bg-transparent p-10 pt-16 bg-gradient-to-t from-bg-light dark:from-bg-dark from-5% via-bg-light dark:via-bg-dark via-10% to-transparent to-100%">
                </div>
            </div>
            <div class="w-full flex flex-row items-center justify-center h-10 mx-auto" v-if="currentDiscussion.id">

                <ChatBox ref="chatBox" 
                    :loading="isGenerating" 
                    :discussionList="discussionArr" 
                    :on-show-toast-message="showToastMessage"
                    :on-talk="talk"

                    @personalitySelected="recoverFiles"
                    @messageSentEvent="sendMsg" 
                    @sendCMDEvent="sendCmd"
                    @addWebLink="add_webpage"
                    @createEmptyUserMessage="createEmptyUserMessage"
                    @createEmptyAIMessage="createEmptyAIMessage"
                    @stopGenerating="stopGenerating" 
                    @loaded="recoverFiles"
                    >
                </ChatBox>
            </div>        
        </div>
    <transition name="slide-left">
    <div  v-if="showRightPanel"
        class="relative flex flex-col no-scrollbar shadow-lg w-1/2 bg-bg-light-tone dark:bg-bg-dark-tone h-full"
        >
        <!-- RIGHT SIDE PANEL -->
         <!--  <div v-html="lastMessageHtml"></div> --> 
        <div ref="isolatedContent" class="h-full"></div>
    </div>
    </transition>
    <ChoiceDialog reference="database_selector" class="z-20"
      :show="database_selectorDialogVisible"
      :choices="databases"
      :can-remove=true      
      @choice-removed="ondatabase_selectorDialogRemoved"
      @choice-selected="ondatabase_selectorDialogSelected"
      @close-dialog="onclosedatabase_selectorDialog"
      @choice-validated="onvalidatedatabase_selectorChoice"
    />      
    <div v-show="progress_visibility" role="status" class="fixed m-0 p-2 left-2 bottom-2 min-w-[24rem] max-w-[24rem] h-20 flex flex-col justify-center items-center pb-4 bg-blue-500 rounded-lg shadow-lg z-50 background-a">
    <ProgressBar ref="progress" :progress="progress_value" class="w-full h-4"></ProgressBar>
    <p class="text-2xl animate-pulse mt-2 text-white">{{ loading_infos }} ...</p>
    </div>     
    <PersonalityEditor ref="personality_editor" :config="currentPersonConfig" :personality="selectedPersonality"></PersonalityEditor>
    <div id="app">
    <PopupViewer ref="news"/>    
    </div>    
    
    <SkillsLibraryViewer ref="skills_lib" ></SkillsLibraryViewer>
    <ChangelogPopup/>
</template>


<style scoped>
.personalities-container {
  position: relative;
}

.skills-lib-icon {
  position: absolute;
  top: -2px;
  right: -2px;
  z-index: 10;
  font-size: 5px; /* Adjust this value to change the icon size */
  width: 2px; /* Match this to the font-size */
  height: 2px; /* Match this to the font-size */
}

.skills-lib-icon i {
  width: 2px;
  height: 2px;
}
@keyframes giggle {
    0%, 100% {
        transform: translateX(0) rotate(0deg) scale(1);
    }
    25% {
        transform: translateX(-5px) rotate(-10deg) scale(1.05);
    }
    50% {
        transform: translateX(5px) rotate(10deg) scale(0.95);
    }
    75% {
        transform: translateX(-5px) rotate(-10deg) scale(1.05);
    }
}

.animate-giggle {
    animation: giggle 1.5s infinite ease-in-out;
}

.custom-scrollbar {
scrollbar-width: thin;
scrollbar-color: rgba(155, 155, 155, 0.5) transparent;
}

.custom-scrollbar::-webkit-scrollbar {
width: 8px;
}

.custom-scrollbar::-webkit-scrollbar-track {
background: transparent;
}

.custom-scrollbar::-webkit-scrollbar-thumb {
background-color: rgba(155, 155, 155, 0.5);
border-radius: 20px;
border: transparent;
}
@keyframes custom-pulse {
    0%, 100% { box-shadow: 0 0 0 0 rgba(59, 130, 246, 0.5); }
    50% { box-shadow: 0 0 0 15px rgba(59, 130, 246, 0); }
  }
.animate-pulse {
animation: custom-pulse 2s infinite;
}

.slide-right-enter-active {
  transition: transform 0.3s ease;
}

.slide-right-leave-active {
  transition: transform 0.3s ease;
}

.slide-right-enter,
.slide-right-leave-to {
  transform: translateX(-100%);
}

.slide-left-enter-active {
  transition: transform 0.3s ease;
}

.slide-left-leave-active {
  transition: transform 0.3s ease;
}

.slide-left-enter,
.slide-left-leave-to {
  transform: translateX(100%);
}

.fade-and-fly-enter-active {
  animation: fade-and-fly-enter 0.5s ease;
}

.fade-and-fly-leave-active {
  animation: fade-and-fly-leave 0.5s ease;
}

@keyframes fade-and-fly-enter {
  0% {
    opacity: 0;
    transform: translateY(20px) scale(0.8);
  }
  100% {
    opacity: 1;
    transform: translateY(0) scale(1);
  }
}

@keyframes fade-and-fly-leave {
  0% {
    opacity: 1;
    transform: translateY(0) scale(1);
  }
  100% {
    opacity: 0;
    transform: translateY(-20px) scale(1.2);
  }
}

/* THESE ARE FOR TransitionGroup components */
.list-move,
/* apply transition to moving elements */
.list-enter-active,
.list-leave-active {
    transition: all 0.5s ease;
}

.list-enter-from {
    transform: translatey(-30px);
}

.list-leave-to {
    opacity: 0;
    transform: translatey(30px);
}

/* ensure leaving items are taken out of layout flow so that moving
   animations can be calculated correctly. */
.list-leave-active {
    position: absolute;
}
@keyframes float {
  0%, 100% { transform: translateY(0); }
  50% { transform: translateY(-20px); }
}
.animate-float {
  animation: float linear infinite;
}

@keyframes star-move {
  0% { transform: translate(0, 0) rotate(0deg); }
  50% { transform: translate(20px, 20px) rotate(180deg); }
  100% { transform: translate(0, 0) rotate(360deg); }
}

.animate-star {
  animation: star-move linear infinite;
}

@keyframes fall {
  from {
    transform: translateY(-20px) rotate(0deg);
    opacity: 1;
  }
  to {
    transform: translateY(calc(100vh + 20px)) rotate(360deg);
    opacity: 0;
  }
}

.animate-fall {
  animation: fall linear infinite;
}

@keyframes glow {
  0%, 100% { 
    text-shadow: 0 0 5px rgba(66, 153, 225, 0.5), 0 0 10px rgba(66, 153, 225, 0.5);
  }
  50% { 
    text-shadow: 0 0 20px rgba(66, 153, 225, 0.8), 0 0 30px rgba(66, 153, 225, 0.8);
  }
}

.animate-glow {
  animation: glow 2s ease-in-out infinite;
}

@media (prefers-color-scheme: dark) {
  @keyframes glow {
    0%, 100% { 
      text-shadow: 0 0 5px rgba(147, 197, 253, 0.5), 0 0 10px rgba(147, 197, 253, 0.5);
    }
    50% { 
      text-shadow: 0 0 20px rgba(147, 197, 253, 0.8), 0 0 30px rgba(147, 197, 253, 0.8);
    }
  }
}

@media (prefers-color-scheme: dark) {
  @keyframes glow {
    0%, 100% { 
      text-shadow: 0 0 5px rgba(147, 197, 253, 0.5), 0 0 10px rgba(147, 197, 253, 0.5);
    }
    50% { 
      text-shadow: 0 0 20px rgba(147, 197, 253, 0.8), 0 0 30px rgba(147, 197, 253, 0.8);
    }
  }
}


@keyframes roll {
  0% {
    transform: translateX(-50%) rotate(0deg);
  }
  100% {
    transform: translateX(50%) rotate(360deg);
  }
}

.animate-roll {
  animation: roll 4s linear infinite;
}


.toolbar {
@apply relative w-full;
}

.toolbar-container {
@apply flex items-center h-10; /* h-10 is equivalent to 40px */
}

.toolbar-button {
@apply bg-transparent border-none cursor-pointer p-2 transition-colors duration-300;
}

.toolbar-button:hover {
@apply text-blue-500; /* Assuming #007bff is close to Tailwind's blue-500 */
}

.menu-container {
  position: relative;
}
.expandable-menu {
  @apply absolute top-full left-2.5 border border-gray-300 rounded flex-col shadow-md;
}


.menu-container:hover .expandable-menu,
.menu-visible {
  display: flex;
}

.menu-item {
  background: none;
  border: none;
  cursor: pointer;
  padding: 8px;
  color: #333;
  transition: background-color 0.3s;
}

.menu-item:hover {
  background-color: #f0f0f0;
}


.dot {
width: 10px;
height: 10px;
border-radius: 50%;
}

.dot-green {
background-color: green;
}

.dot-red {
background-color: red;
}

.animate-pulse {
animation: pulse 2s cubic-bezier(0.4, 0, 0.6, 1) infinite;
}

@keyframes pulse {
0%, 100% {
    opacity: 1;
}
50% {
    opacity: .7;
}
}
.logo-container {
position: relative;
width: 48px;
height: 48px;
}

.logo-image {
width: 100%;
height: 100%;
border-radius: 50%;
object-fit: cover;
}
@keyframes bounce {
0%, 100% {
    transform: translateY(-25%);
    animation-timing-function: cubic-bezier(0.8, 0, 1, 1);
}
50% {
    transform: translateY(0);
    animation-timing-function: cubic-bezier(0, 0, 0.2, 1);
}
}
.animate-bounce {
animation: bounce 1s infinite;
}


@keyframes roll-and-bounce {
0%, 100% {
    transform: translateX(0) rotate(0deg);
}
45% {
    transform: translateX(100px) rotate(360deg);
}
50% {
    transform: translateX(90px) rotate(390deg);
}
55% {
    transform: translateX(100px) rotate(360deg);
}
95% {
    transform: translateX(0) rotate(0deg);
}
}

@keyframes spin {
0% {
    transform: rotate(0deg);
}
100% {
    transform: rotate(360deg);
}
}
</style>
<script>
import PersonalityEditor from "@/components/PersonalityEditor.vue"
import PopupViewer from '@/components/PopupViewer.vue';
import ActionButton from '@/components/ActionButton.vue'
import SocialIcon from '@/components/SocialIcon.vue'

import FastAPI from '@/assets/fastapi.png';
import discord from '@/assets/discord.svg';
import { RouterLink } from 'vue-router'
import { nextTick } from 'vue'
import feather from 'feather-icons'

import static_info from "../assets/static_info.svg"
import animated_info from "../assets/animated_info.svg"
import { useRouter } from 'vue-router'
import storeLogo from '@/assets/logo.png'
import fun_mode from "../assets/fun_mode.svg"
import normal_mode from "../assets/normal_mode.svg"

import axios from 'axios';
import { store } from '../main';

import SVGRedBrain from '@/assets/brain_red.svg';
import SVGOrangeBrain from '@/assets/brain_orange.svg';
import SVGGreenBrain from '@/assets/brain_green.svg';
import memory_icon from "../assets/memory_icon.svg"
import active_skills from "../assets/active.svg"
import inactive_skills from "../assets/inactive.svg"
import skillsRegistry from "../assets/registry.svg"
import robot from "../assets/robot.svg"
import { mapState } from 'vuex';
import modelImgPlaceholder from "../assets/default_model.png"

import MountedPersonalities from '@/components/MountedPersonalities.vue'
import ChatBarButton from '@/components/ChatBarButton.vue'
import ChangelogPopup from "@/components/ChangelogPopup.vue"


const parsePlaceholder = (placeholder) => {
    const parts = placeholder.replace('[', '').replace(']', '').split('::');
    const label = parts[0];

    if (parts.length === 1) {
        return {
            label,
            type: 'text',
            fullText: placeholder
        };
    }

    const type = parts[1];
    const result = {
        label,
        type,
        fullText: placeholder
    };

    switch (type) {
        case 'int':
        case 'float':
        case 'multiline':
            break;
        case 'code':
            result.language = parts[2] || 'plaintext';
            break;
        case 'options':
            result.options = parts[2] ? parts[2].split(',').map(o => o.trim()) : [];
            break;
        default:
            result.type = 'text';
    }
    return result;
};

const bUrl = import.meta.env.VITE_LOLLMS_API_BASEURL
export default {
    
    setup() { },
    
    data() {
        return {
            interestingFacts: [
                "ParisNeo, the creator of LoLLMs, originally built his high-performance PC to play Cyberpunk 2077. However, his passion for AI took an unexpected turn, leading him to develop LoLLMs instead. Ironically, he never found the time to actually play the game that inspired his powerful setup!",
                "Saïph, version 14 of LoLLMs, is named after a star in Orion's constellation (Kappa Orionis), representing bright guidance in AI!",
                "At 12, ParisNeo coded 'Saif14' in BASIC on a Tandy 1000, an 'all-in-one' suite named with a friend. LoLLMs v14 'Saïph' - a star in Orion - echoes this BASIC creation's spirit and hints at OpenAI's future.",
                "The new AI Village app in LoLLMs, simulating AI agents believing they're human, mirrors \"The Hitchhiker's Guide to the Galaxy\" (ParisNeo's favorite) where Earth was a simulation to compute the Ultimate Question. Both explore the idea of unsuspecting inhabitants participating in a grand cosmic calculation!",
                "Once upon a time, young ParisNeo dove so deep into \"The Matrix\" that he watched it 60 times in two months, practically becoming a human DVD player! Fast forward to today, and this same Matrix-obsessed kid grew up to create LoLLMs, where AI personas are like digital kung-fu masters, and users are the chosen ones. Talk about life imitating art - ParisNeo went from \"I know kung fu\" to \"I code AI\"! Who knew binge-watching could lead to bending the rules of AI reality?",
                "Did you know? The first computer programmer was a woman - Ada Lovelace!",
                "Did you know? The term 'bug' in computing came from an actual moth trapped in a computer relay in 1947!",
                "The first computer mouse was made of wood in the 1960s by Doug Engelbart!",
                "The entire Apollo 11 mission that landed humans on the moon used less computing power than today's average smartphone!",
                "Did you know? The first emoji was created in 1999 by Shigetaka Kurita in Japan!",
                "The world's first website, created by Tim Berners-Lee, went live on August 6, 1991, and it's still online today!",
                "Did you know? A neural network once learned to detect tanks in photos by accidentally learning to detect the time of day - all tank photos were taken during daytime!",
                "The first computer virus was created in 1971 and was called 'Creeper'. It simply displayed the message 'I'm the creeper: catch me if you can!'",
                "Did you know? The CAPTCHA system you use to prove you're human actually helps digitize books and improve AI training!",
                "The word 'robot' comes from the Czech word 'robota', meaning forced labor or servitude!",
                "Did you know? The first AI chatbot, ELIZA, was created in 1966 and could convince some people they were talking to a real therapist!",
                "Large Language Models (LLMs) have evolved from having millions of parameters to hundreds of billions in just a few years.",
                "LoLLMs (Lord of Large Language Multimodal Systems) is an open-source AI assistant platform created by ParisNeo.",
                "Saïph (κ Orionis) is a blue-white supergiant star approximately 650 light-years away from Earth.",
                "Neural networks were first proposed in 1943 by Warren McCulloch and Walter Pitts.",
                "Modern LLMs like GPT-4 can understand and generate multiple languages, code, and even analyze images.",
                "LoLLMs supports multiple AI models and can perform tasks like code interpretation, image analysis, and internet searches.",
                "The term 'transformer' in AI, which powers most modern LLMs, was introduced in the 'Attention is All You Need' paper in 2017.",
                "LoLLMs can generate various types of diagrams, including SVG, Graphviz, and Mermaid diagrams.",
                "The Python programming language was named after Monty Python.",
                "LoLLMs features a built-in code interpreter that can execute multiple programming languages.",
                "Quantum computers can perform calculations in minutes that would take classical computers thousands of years.",
                "LoLLMs supports multimodal interactions, allowing users to work with both text and images.",
                "Did you know? Honeybees build their honeycombs in perfect hexagons, the most efficient shape for storing honey!",
                "The ancient Egyptians used a simple rope with 12 knots to create perfect right angles for building pyramids!",
                "Did you know? There's a shape called a 'mobius strip' that has only one side and one edge - try making one with a paper strip!",
                "The sum of angles in any triangle is always 180 degrees, whether it's tiny or huge!",
                "Did you know? A pizza slice demonstrates the mathematical concept of sectors and radians - math you can eat!",
                "The ancient Greeks believed there were only 5 'perfect' 3D shapes, called Platonic solids!",
                "Did you know? Fractals are shapes that repeat infinitely - they're found everywhere in nature, from snowflakes to broccoli!",
                "A circle has an infinite number of lines of symmetry, more than any other shape!",
                "Did you know? The Golden Ratio (approximately 1.618) appears in nature and is considered the most aesthetically pleasing proportion!",
                "The shortest path between two points on a sphere isn't a straight line - it's called a 'great circle' route, used by airlines!",
                "Did you know? The first computer virus wasn't created to cause harm - it was a game called 'Core War' where programs fought for control of the computer's memory!",
                "A single Google search uses more computing power than the entire Apollo space mission!",
                "Did you know? The world's first programmer, Ada Lovelace, wrote an algorithm in 1842 before computers even existed!",
                "The average smartphone today is millions of times more powerful than all of NASA's combined computing power in 1969!",
                "Did you know? The first computer 'bug' was literally a moth trapped in a relay of the Mark II computer in 1947!",
                "Bitcoin's entire source code is shorter than Microsoft Word's printer drivers!",
                "Did you know? The most expensive typing error in history occurred in 1962 when NASA's Mariner 1 spacecraft was destroyed due to a missing hyphen in its code!",
                "The first computer mouse was made of wood and had two wheels instead of a ball or optical sensor!",
                "Did you know? The entire internet weighs about as much as a strawberry - if you were to weigh all its electrons!",
                "The popular programming language Python wasn't named after the snake - it was named after Monty Python's Flying Circus!",
                "Did you know? The first 1GB hard drive released in 1980 weighed about 550 pounds and cost $40,000!",
                "The QWERTY keyboard layout was designed to slow typists down to prevent mechanical typewriters from jamming!",
                "The name Saïph in Arabic (سيف) means 'sword', symbolizing cutting-edge AI technology.",
                '<div class="flex items-center justify-center"><iframe width="560" height="315" src="https://www.youtube.com/embed/7pSXGj0dSzE?si=Ov0Y4F2mCRSB61xc" title="YouTube video player" frameborder="0" allow="accelerometer; autoplay; clipboard-write; encrypted-media; gyroscope; picture-in-picture; web-share" referrerpolicy="strict-origin-when-cross-origin" allowfullscreen></iframe></div>',
                "LoLLMs' version naming often contains clever easter eggs and references to AI advancements.",
                "The 'Strawberry' version of LoLLMs was a playful nod to ChatGPT's internal codename for one of its versions.",
                "The 'Saïph' version name was an intentional reference to Orion, anticipating OpenAI's rumored AGI-capable model codenamed 'Orion'.",
                "LoLLMs' evolution can be traced through its version names: Warp, Starship, Robot, Brainwave, Strawberry, Feather, Saïph, Nexus, Pulsar.",
                "Each LoLLMs version name reflects either technological advancement or pays homage to significant developments in AI.",
                "'Warp' and 'Starship' versions symbolized the quantum leap in AI capabilities and speed improvements.",
                "'Robot' represented the system's growing autonomy and ability to perform complex tasks.",
                "'Brainwave' highlighted the neural network aspects and cognitive capabilities of the system.",
                "LoLLMs' version naming shows ParisNeo's keen awareness of industry trends and playful approach to development.",   
                "LoLLMs can generate and visualize mathematical equations using LaTeX, making it a powerful tool for scientific documentation.",
                "The system's multimodel capabilities allow it to analyze medical images, architectural blueprints, and technical diagrams.",
                "LoLLMs includes a unique feature called 'personality system' that allows it to adapt its communication style and expertise.",
                "Did you know? LoLLMs can process and generate music notation using ABC notation or LilyPond formats.",
                "LoLLMs supports over 2000 different AI models, making it one of the most versatile open-source AI platforms.",
                "The system can generate realistic 3D scenes descriptions that can be rendered using tools like Blender.",
                "Did you know? One day on Venus is longer than its year - it takes 243 Earth days to rotate on its axis but only 225 days to orbit the Sun!",
                "The footprints on the Moon will last for 100 million years because there is no wind to blow them away!",
                "Did you know? If you could put Saturn in a giant bathtub, it would float - it's the only planet less dense than water!",
                "The Great Red Spot on Jupiter is a storm that has been raging for at least 400 years!",
                "Did you know? There's a planet made largely of diamonds, called 55 Cancri e - it's worth about $26.9 nonillion!",
                "A day on the International Space Station has 16 sunrises and sunsets because it orbits Earth every 90 minutes!",
                "Did you know? The largest known star, UY Scuti, is so big that it would take 1,700 years to walk around it at walking speed!",
                "There's a cloud of alcohol in space - Sagittarius B2 contains billions of liters of ethyl alcohol!",
                "Did you know? The sound of a black hole is actually a B-flat note, 57 octaves below middle C!",
                "Space isn't completely silent - planets and stars emit electromagnetic vibrations that can be converted into sound!",
                "Did you know? If you cry in space, your tears just stick to your face - they won't fall because there's no gravity!",
                "The entire human race could fit inside a sugar cube if all the empty space in our atoms was removed!",
                "Did you know? The green code in The Matrix isn't actually computer code - it's sushi recipes written in Japanese characters!",
                "The Matrix used approximately 12,000 pounds of concrete to create the iconic lobby scene destruction!",
                "Did you know? Keanu Reeves had to undergo spinal surgery before filming The Matrix Reloaded due to neck injuries from the first movie's stunts!",
                "The famous 'bullet time' effect required 120 cameras shooting simultaneously and took months to create just a few seconds of footage!",
                "Did you know? The actors had to train for 4 months in martial arts, wire work, and fight choreography before filming began!",
                "The color green was deliberately used throughout the movie to create a connection to the old computer monitors of the 1980s!",
                "Did you know? Will Smith turned down the role of Neo to star in Wild Wild West - a decision he later regretted!",
                "The sunglasses worn in The Matrix were custom-made and cost over $2,000 per pair!",
                "Did you know? The scene where Neo sees the cat twice was actually inspired by a programming concept called déjà vu in computer simulation!",
                "The Matrix's original budget was $60 million, but the opening sequence alone cost $10 million to create!",
                "Did you know? The red pill and blue pill were actually made of minted candy to ensure they looked perfect in every take!",
                "The famous dojo fight scene between Neo and Morpheus took 6 days to film and both actors performed most of their own stunts!",
                "LoLLMs features a unique 'model fusion' capability, combining strengths of different AI models for better results.",
                "The platform includes specialized modules for scientific computing, allowing it to solve complex mathematical problems.",
                "LoLLMs can analyze and generate code in over 20 programming languages, including rare ones like COBOL and Fortran.",
                "The system includes advanced prompt engineering tools, helping users get better results from AI models.",
                "LoLLMs can generate and interpret QR codes, making it useful for creating interactive marketing materials.",
                "The platform supports real-time voice interaction through its advanced speech-to-text and text-to-speech capabilities.",
                "LoLLMs can analyze satellite imagery for environmental monitoring and urban planning applications.",
                "The system includes specialized modules for protein folding prediction and molecular visualization.",
                "LoLLMs features a built-in 'ethical AI' framework that ensures responsible and bias-aware AI interactions.",
                "The platform can generate realistic synthetic data while preserving privacy and maintaining statistical properties.",
                "LoLLMs includes advanced natural language processing capabilities in over 100 languages.",
                "The system can perform sentiment analysis on social media trends and customer feedback in real-time.",
                "LoLLMs features a unique 'time-aware' context system that understands and reasons about temporal relationships.",
                "The world's first computer programmer was a woman named Ada Lovelace.",
                "There are more possible iterations of a game of chess than there are atoms in the known universe.",
                "The first AI program was written in 1951 to play checkers.",
                "The Internet was originally called ARPANET and was developed by the US Department of Defense.",
                "The concept of zero as a number was developed in ancient India, with the earliest known text using it dating to 458 CE.",
                "Light from the Sun takes about 8 minutes and 20 seconds to reach Earth.",
                "The first mobile phone call was made in 1973 by Martin Cooper.",
                "The human brain processes images 60,000 times faster than text.",
                "The most powerful supercomputer today can perform about 442 quadrillion calculations per second.",
                "The ancient Egyptians invented the 365-day calendar we still use today.",
                "Mathematicians have calculated pi to over 62 trillion digits.",
                "Quantum computers can theoretically solve certain problems in seconds that would take classical computers billions of years.",
                "The first email was sent in 1971 by Ray Tomlinson.",
                "The Hubble Space Telescope can see galaxies so far away, it's like looking back in time billions of years.",
                "The largest known prime number has over 24 million digits.",
                "AI can now generate realistic images, music, and even write stories.",
                "The Great Wall of China is not visible from space with the naked eye, contrary to popular belief.",
                "There are more possible ways to shuffle a deck of 52 cards than there are atoms on Earth.",
                "The first computer virus was created in 1983 as an experiment.",
                "The speed of light in a vacuum is exactly 299,792,458 meters per second.",
                "In version 18 (Matrix), the changelog contained a binary secret message from ParisNeo.\nDid you get it?\nIt meant THERE IS NO SPOON, the matirx's famous scene.",
                "ParisNeo started coding at age 11 after an older programmer refused to teach him BASIC programming",
                "ParisNeo learned BASIC by reverse engineering other people's games, even before he learned English",
                "At age 12, ParisNeo co-created SAIF14, a suite of programs on a Tandy 1000 computer built by two kids",
                "LoLLMs runs on a custom-built gaming PC that ParisNeo assembled and configured himself",
                "The project started as a simple Chrome plugin called 'chatgpt personality selector'",
                "ParisNeo's summer bet involved watching The Matrix 60 times in one month (twice daily) while his uncle gave up",
                "The name LoLLMs was inspired by 'Lord of the Rings', positioning itself as 'one tool to rule them all'",
                "LoLLMs has a 'Personalities Zoo' with over 500 different AI personas",
                "The project evolved from handling just language models to becoming a multimodal system, hence the 'M' in LoLLMs",
                "ParisNeo's difficulty with academic writing led him to develop AI solutions for writing assistance",
                "LoLLMs' development was influenced by science fiction concepts from Wall-E, Terminator, and The Hitchhiker's Guide to the Galaxy",
                "The project went from being 'GPT4All WebUI' to 'Lord of Large Language Models' before becoming 'Lord of Large Language & Multimodal Systems'",
                "ParisNeo received early access to GPT-3 and used it to push the boundaries of what was possible with AI",
                "The project maintains a philosophy of 'Don't panic, bring a towel, and trust in AI' - a reference to The Hitchhiker's Guide to the Galaxy",
                "Despite having multiple patents in various fields, ParisNeo's interest in AI was sparked when he read about AlexNet in 2012",
                "LoLLMs receives daily updates, similar to course corrections on a starship",
                "Fun Fact: Neo's apartment number 101 in The Matrix is a brilliant multi-layered reference! In binary, 101 equals 5, representing someone who stands 'apart' from the zeros and ones of the Matrix. But there's more - Room 101 is also famous from Orwell's '1984' as the room containing 'the worst thing in the world.' For Neo, his apartment 101 was indeed where he faced his worst fear: discovering his entire reality was a lie. ParisNeo slyly references this in LoLLMs v18 by making '101' a special command that reveals the system's true capabilities!",
                "ParisNeo's journey into programming shows his determination - he taught himself BASIC by reverse engineering games before he even knew English",
                "Version 17 of LoLLMs, nicknamed 'Pulsar', draws inspiration from the most precise natural clocks in the universe - pulsating neutron stars. Just as pulsars emit precise, regular signals that help navigate spacecraft, LoLLMs Pulsar orchestrates precise communication between different AI systems. The version number 17 wasn't just sequential - pulsars rotate up to 1.7 thousand times per second, making '17' a cosmic nod to these incredible stellar lighthouses of space!",
                "Version 18 'Matrix' of LoLLMs is a delightful nod to ParisNeo's teenage obsession with The Matrix movie - talk about coming full circle from watching the film 60 times to creating an AI system that lets users 'bend the rules' of reality!",
                "Did you know? The '18' in LoLLMs v18 'Matrix' has a hidden meaning - the '1' represents The One (like Neo), while the '8' represents infinity (∞) when turned sideways. Just as Neo broke free from the infinite cycle of the Matrix, LoLLMs v18 breaks traditional AI limitations!",
                "LoLLMs v18 Matrix introduces a groundbreaking feature called 'Digital Twins' - multiple AI personas that can interact with each other, much like the multiplying Agent Smith in The Matrix Reloaded. Life imitating art at its finest!",
                "When ParisNeo announced version 18 'Matrix', he included a hidden binary message in the release notes that, when decoded, read 'There is no spoon' - a playful reference to both The Matrix and the flexible nature of AI reality.",
                "The Matrix version introduced 'Morpheus Mode' - an advanced teaching capability that guides users through complex AI concepts, just like Morpheus guided Neo through the nature of the Matrix. 'Unfortunately, no one can be told what LoLLMs is. You have to experience it for yourself.'",
                "In a cheeky nod to The Matrix's famous green code rain, LoLLMs v18 includes an easter egg where typing 'follow the white rabbit' triggers a special visualization mode for code generation.",
                "While previous versions were inspired by celestial bodies (like Saïph and Pulsar), Matrix marks LoLLMs' first version named after a simulation concept - reflecting the growing capabilities of AI to create and manage virtual worlds.",
                "The development of Matrix version started exactly 18 years after ParisNeo's infamous Matrix-watching marathon, making this version a personal milestone in his journey from movie fan to AI innovator.",
                "In the Matrix version's documentation, ParisNeo included a special section titled 'What is the Matrix?' that explains complex AI concepts using analogies from the movie, making advanced technology accessible to everyone - just like Morpheus did for Neo!",
                "Did you know? Chuck Norris counted to infinity... twice!",
                "When Chuck Norris does pushups, he doesn't push himself up - he pushes the Earth down!",
                "Did you know? Chuck Norris's tears can cure cancer... too bad he has never cried!",
                "Chuck Norris has a grizzly bear carpet in his room. The bear isn't dead, it's just afraid to move!",
                "Did you know? Chuck Norris was actually born in Ryan, Oklahoma, and his real name is Carlos Ray Norris!",
                "When Chuck Norris enters a room, he doesn't turn the lights on - he turns the dark off!",
                "Did you know? Chuck Norris is the only person who can slam a revolving door!",
                "Chuck Norris doesn't wear a watch. He decides what time it is!",
                "Did you know? Chuck Norris actually served in the United States Air Force as an Air Policeman in South Korea!",
                "When the Boogeyman goes to sleep every night, he checks his closet for Chuck Norris!",
                "Did you know? Chuck Norris holds the record for the most consecutive push-ups without rest at 5,000!",
                "Time waits for no man. Unless that man is Chuck Norris!"                
            ],
            currentVersion: null,
            showChangelogPopup: false,
            randomFact: "",            
            showPlaceholderModal: false,
            selectedPrompt: '',
            placeholders: [],
            placeholderValues: {},
            previewPrompt: '',
            uniquePlaceholders: new Map(),

            bindingSearchQuery: '',
            modelSearchQuery: '',
            personalitySearchQuery: '',
            isSearching: false,
            isPersonalitiesMenuVisible: false,
            isSkillsLibraryMenuVisible: false,
            isModelsMenuVisible:false,
            isBindingsMenuVisible: false,
            isMenuVisible: false,
            isNavMenuVisible: false,
            static_info: static_info,
            animated_info: animated_info,
            normal_mode:normal_mode,
            fun_mode:fun_mode,
            is_first_connection:true,
            discord:discord,
            FastAPI:FastAPI,
            modelImgPlaceholder:modelImgPlaceholder,
            customLanguage: '', // Holds the value of the custom language input
            rebooting_audio: new Audio("rebooting.wav"),            
            connection_lost_audio: new Audio("connection_lost.wav"),
            connection_recovered_audio: new Audio("connection_recovered.wav"),
            database_selectorDialogVisible:false,
            progress_visibility:false,
            progress_value:0,
            codeBlockStylesheet:'',

            lastMessageHtml:"",
            defaultMessageHtml: `
        <!DOCTYPE html>
        <html lang="en">
        <head>
            <meta charset="UTF-8">
            <meta name="viewport" content="width=device-width, initial-scale=1.0">
            <title>Default Render Panel</title>
            <style>
                body, html {
                    margin: 0;
                    padding: 0;
                    height: 100%;
                    font-family: Arial, sans-serif;
                    background-color: #f0f0f0;
                }
                .container {
                    display: flex;
                    flex-direction: column;
                    align-items: center;
                    justify-content: center;
                    height: 100%;
                    padding: 20px;
                    box-sizing: border-box;
                }
                .message {
                    text-align: center;
                    padding: 30px;
                    background-color: white;
                    border-radius: 12px;
                    box-shadow: 0 4px 20px rgba(0,0,0,0.1);
                    max-width: 600px;
                    width: 100%;
                }
                h1 {
                    color: #2c3e50;
                    margin-bottom: 20px;
                    font-size: 28px;
                }
                p {
                    color: #34495e;
                    margin: 0 0 15px;
                    line-height: 1.6;
                }
                .highlight {
                    color: #e74c3c;
                    font-weight: bold;
                }
            </style>
        </head>
        <body>
            <div class="container">
                <div class="message">
                    <h1>Welcome to the Interactive Render Panel</h1>
                    <p>This space is designed to bring your ideas to life! Currently, it's empty because no HTML has been generated yet.</p>
                    <p>To see something amazing here, try asking the AI to <span class="highlight">create a specific web component or application</span>. For example:</p>
                    <p>"Create a responsive image gallery" or "Build a simple todo list app"</p>
                    <p>Once you request a web component, the AI will generate the necessary HTML, CSS, and JavaScript, and it will be rendered right here in this panel!</p>
                </div>
            </div>
        </body>
        </html>       
            `,
            memory_icon: memory_icon,
            active_skills:active_skills,
            inactive_skills:inactive_skills,
            skillsRegistry:skillsRegistry,
            robot:robot,
            posts_headers : {
                'accept': 'application/json',
                'Content-Type': 'application/json'
            },
            host:"",
            progress_visibility_val         : true,
            progress_value                  : 0,
            msgTypes: {
                // Messaging
                MSG_TYPE_CONTENT                   : 1, // A full message (for some personality the answer is sent in bulk)
                MSG_TYPE_CONTENT_INVISIBLE_TO_AI   : 2, // A full message (for some personality the answer is sent in bulk)
                MSG_TYPE_CONTENT_INVISIBLE_TO_USER : 3, // A full message (for some personality the answer is sent in bulk)
            },
            // To be synced with the backend database types
            operationTypes: {
                // Messaging
                MSG_OPERATION_TYPE_ADD_CHUNK    : 0, // Add a chunk to the current message
                MSG_OPERATION_TYPE_SET_CONTENT  : 1, // sets the content of current message
                MSG_OPERATION_TYPE_SET_CONTENT_INVISIBLE_TO_AI      : 2, // sets the content of current message as invisible to ai
                MSG_OPERATION_TYPE_SET_CONTENT_INVISIBLE_TO_USER    : 3, // sets the content of current message as invisible to user
                // Informations
                MSG_OPERATION_TYPE_EXCEPTION              : 4, // An exception occured
                MSG_OPERATION_TYPE_WARNING                : 5, // A warning occured
                MSG_OPERATION_TYPE_INFO                   : 6, // An information to be shown to user

                // Steps
                MSG_OPERATION_TYPE_STEP                   : 7, // An instant step (a step that doesn't need time to be executed)
                MSG_OPERATION_TYPE_STEP_START             : 8, // A step has started (the text contains an explanation of the step done by he personality)
                MSG_OPERATION_TYPE_STEP_PROGRESS          : 9, // The progress value (the text contains a percentage and can be parsed by the reception)
                MSG_OPERATION_TYPE_STEP_END_SUCCESS       : 10, // A step has been done (the text contains an explanation of the step done by he personality)
                MSG_OPERATION_TYPE_STEP_END_FAILURE       : 11, // A step has been done (the text contains an explanation of the step done by he personality)

                //Extra
                MSG_OPERATION_TYPE_JSON_INFOS             : 12, // A JSON output that is useful for summarizing the process of generation used by personalities like chain of thoughts and tree of thooughts
                MSG_OPERATION_TYPE_REF                    : 13, // References (in form of  [text](path))
                MSG_OPERATION_TYPE_CODE                   : 14, // A javascript code to execute
                MSG_OPERATION_TYPE_UI                     : 15, // A vue.js component to show (we need to build some and parse the text to show it)

                //Commands
                MSG_OPERATION_TYPE_NEW_MESSAGE            : 16, // A new message
                MSG_OPERATION_TYPE_FINISHED_MESSAGE       : 17, // End of current message

            },
            // Sender types
            senderTypes: {
                SENDER_TYPES_USER               : 0, // Sent by user
                SENDER_TYPES_AI                 : 1, // Sent by ai
                SENDER_TYPES_SYSTEM             : 2, // Sent by athe system
            },
            discussionsList                                : [], // Discussion list
            tempList                            : [], // Copy of Discussion list (used for keeping the original list during filtering discussions/searching action)
            currentDiscussion                   : {}, // Current/selected discussion id
            discussionArr                       : [],
            loading: false,
            filterTitle: '',
            filterInProgress: false,
            isCreated: false,
            isCheckbox: false,
            isSelectAll: false,
            showSaveConfirmation: false,
            showBrainConfirmation: false,
            showConfirmation: false,
            chime: new Audio("chime_aud.wav"),
            showToast: false,
            isDiscussionBottom: false,
            personalityAvatars: [], // object array of personality name: and avatar: props
            fileList: [],
            database_selectorDialogVisible:false,
            isDragOverDiscussion: false,
            isDragOverChat: false,
            isOpen: false,
            discussion_id: 0,
        }
    },
    methods: {        
        updateRandomFact() {
            // Get a new random fact different from the current one
            let newFact;
            do {
                newFact = this.interestingFacts[Math.floor(Math.random() * this.interestingFacts.length)];
            } while (newFact === this.randomFact && this.interestingFacts.length > 1);
            
            this.randomFact = newFact;
        },        
        async handleOnTalk(){
            const pers = this.mountedPers
            console.log("pers:",pers)
            this.isGenerating = true;
            this.setDiscussionLoading(this.currentDiscussion.id, this.isGenerating);
            let res = await axios.get('/get_generation_status', {})
            if (res) {
                //console.log(res.data.status);
                if (!res.data.status) {
                    const id = this.$store.state.config.personalities.findIndex(item => item === pers.full_path)
                    const obj = {
                    client_id:this.$store.state.client_id,
                    id: id
                    }
                    res = await axios.post('/select_personality', obj);

                    console.log('Generating message from ',res.data.status);
                    socket.emit('generate_msg_from', { id: -1 });
                }
                else {
                    console.log("Already generating");
                }
            }
        },
        onPersonalitiesReadyFun(){
            this.$store.state.personalities_ready = true;
        },
        async showBindingHoveredIn(index){
            this.bindingHoveredIndex = index
        },
        async showBindingHoveredOut(){
            this.bindingHoveredIndex = null
        },

        async showModelHoveredIn(index){
            this.modelHoveredIndex = index
        },
        async showModelHoveredOut(){
            this.modelHoveredIndex = null
        },
        async showPersonalityHoveredIn(index){
            this.personalityHoveredIndex = index
        },
        async showPersonalityHoveredOut(){
            this.personalityHoveredIndex = null
        },
        async onPersonalitySelected(pers) {
            this.hidePersonalitiesMenu()
            // eslint-disable-next-line no-unused-vars
            if (pers) {

                if (pers.selected) {
                    this.$store.state.toast.showToast("Personality already selected", 4, true)
                    return
                }


                //this.settingsChanged = true
                const pers_path = pers.full_path
                if (this.$store.state.config.personalities.includes(pers_path)) {

                    const res = await this.select_personality(pers)
                    await this.$store.dispatch('refreshConfig');    
                    await this.$store.dispatch('refreshBindings');
                    await this.$store.dispatch('refreshModelsZoo');
                    await this.$store.dispatch('refreshModels');
                    await this.$store.dispatch('refreshMountedPersonalities');
                    await this.$store.dispatch('refreshConfig');    
                    await this.$store.dispatch('fetchLanguages');
                    await this.$store.dispatch('fetchLanguage');
                    await this.$store.dispatch('fetchisRTOn');
                    

                    if (res && res.status && res.active_personality_id > -1) {
                        this.$store.state.toast.showToast("Selected personality:\n" + pers.name, 4, true)

                    } else {
                        this.$store.state.toast.showToast("Error on select personality:\n" + pers.name, 4, false)
                    }

                } else {
                }

                this.$emit('personalitySelected')
            

                nextTick(() => {
                    feather.replace()

                })

            }

        },    
        async select_personality(pers) {
            if (!pers) { return { 'status': false, 'error': 'no personality - select_personality' } }
            const pers_path = pers.full_path
            const id = this.$store.state.config.personalities.findIndex(item => item === pers_path)

            const obj = {
                client_id:this.$store.state.client_id,
                id: id
            }


            try {
                const res = await axios.post('/select_personality', obj);

                if (res) {

                    this.$store.dispatch('refreshConfig').then(() => {
                        this.$store.dispatch('refreshPersonalitiesZoo').then(() => {
                        this.$store.dispatch('refreshMountedPersonalities');                
                        });
                    });
                    return res.data

                }
            } catch (error) {
                console.log(error.message, 'select_personality - settings')
                return
            }

        },        
        showPersonalitiesMenu() {
            clearTimeout(this.hideMenuTimeout);
            this.isPersonalitiesMenuVisible = true
        },
        showSkillsLibraryMenu() {
            clearTimeout(this.hideSkillsLibraryMenuTimeout);
            this.isSkillsLibraryMenuVisible = true
        },
        hideSkillsLibraryMenu() {
            this.hideMenuTimeout = setTimeout(() => {
                this.isSkillsLibraryMenuVisible = false;
            }, 300); // 300ms delay before hiding the menu            
        },
        hidePersonalitiesMenu() {
            this.hideMenuTimeout = setTimeout(() => {
                this.isPersonalitiesMenuVisible = false;
            }, 300); // 300ms delay before hiding the menu            
        },
        copyModelName(){
            navigator.clipboard.writeText(this.binding_name + "::" + this.model_name);
            this.$store.state.toast.showToast("Model name copyed to clipboard: "+this.binding_name + "::" + this.model_name, 4, true)
        },
        copyModelNameFrom(model){
            navigator.clipboard.writeText(this.binding_name + "::" + model);
            this.$store.state.toast.showToast("Model name copyed to clipboard: "+this.binding_name + "::" + this.model_name, 4, true)
        },        
        showBindingsMenu() {
            clearTimeout(this.hideBindingsMenuTimeout);
            this.isBindingsMenuVisible = true
        },
        hideBindingsMenu() {
            this.hideBindingsMenuTimeout = setTimeout(() => {
                this.isBindingsMenuVisible = false;
            }, 300); // 300ms delay before hiding the menu            
        },        
        setBinding(selectedBinding){
            this.selecting_binding=true
            this.selectedBinding = selectedBinding
            this.$store.state.messageBox.showBlockingMessage("Loading binding")

            axios.post("/update_setting", {    
                        client_id: this.$store.state.client_id,
                        setting_name: "binding_name",
                        setting_value: selectedBinding.name
                    }).then(async (response) => {
                this.$store.state.messageBox.hideMessage()
                await this.$store.dispatch('refreshConfig');    
                await this.$store.dispatch('refreshBindings');
                await this.$store.dispatch('refreshModelsZoo');
                await this.$store.dispatch('refreshModels');
                
                this.$store.state.toast.showToast(`Binding changed to ${this.currentBinding.name}`,4,true)
                this.selecting_binding=false
                }).catch(err=>{
                this.$store.state.messageBox.hideMessage()
                this.$store.state.toast.showToast(`Error ${err}`,4,true)
                this.selecting_binding=false
                });            
        },
        showModelsMenu() {
            clearTimeout(this.hideModelsMenuTimeout);
            this.isModelsMenuVisible = true
        },
        hideModelsMenu() {
            this.hideModelsMenuTimeout = setTimeout(() => {
                this.isModelsMenuVisible = false;
            }, 300); // 300ms delay before hiding the menu            
        },

        setModel(selectedModel){
            this.selecting_model=true
            this.selectedModel = selectedModel
            this.$store.state.messageBox.showBlockingMessage("Loading model")
            axios.post("/update_setting", {     
                        client_id: this.$store.state.client_id,           
                        setting_name: "model_name",
                        setting_value: selectedModel.name
                    }).then(async (response) => {
                this.$store.state.messageBox.hideMessage()
                await this.$store.dispatch('refreshConfig');    
                await this.$store.dispatch('refreshModels');
                this.$store.state.toast.showToast(`Model changed to ${this.currentModel.name}`,4,true)
                this.selecting_model=false
                }).catch(err=>{
                this.$store.state.messageBox.hideMessage()
                this.$store.state.toast.showToast(`Error ${err}`,4,true)
                this.selecting_model=false
                });
        
        },        
        showModelConfig(){
            try {
                this.isLoading = true
                axios.get('/get_active_binding_settings').then(res => {
                    this.isLoading = false
                    if (res) {
                        if (res.data && Object.keys(res.data).length > 0) {

                            // open form

                            this.$store.state.universalForm.showForm(res.data, "Binding settings ", "Save changes", "Cancel").then(res => {
                                // send new data
                                try {
                                    axios.post('/set_active_binding_settings',
                                    {client_id:this.$store.state.client_id, "settings":res}).then(response => {

                                            if (response && response.data) {
                                                this.$store.state.toast.showToast("Binding settings updated successfully!", 4, true)

                                            } else {
                                                this.$store.state.toast.showToast("Did not get binding settings responses.\n" + response, 4, false)
                                                this.isLoading = false
                                            }


                                        })
                                } catch (error) {
                                    this.$store.state.toast.showToast("Did not get binding settings responses.\n Endpoint error: " + error.message, 4, false)
                                    this.isLoading = false
                                }



                            })
                        } else {
                            this.$store.state.toast.showToast("Binding has no settings", 4, false)
                            this.isLoading = false
                        }

                    }
                })

            } catch (error) {
                this.isLoading = false
                this.$store.state.toast.showToast("Could not open binding settings. Endpoint error: " + error.message, 4, false)
            }
        },
        async remount_personality(pers) {
            if (!pers) { return { 'status': false, 'error': 'no personality - mount_personality' } }
            try {
                const obj = {
                    client_id: this.$store.state.client_id,
                    category: pers.category,
                    folder: pers.folder,
                    language: pers.language
                }
                const res = await axios.post('/remount_personality', obj);
                

                if (res) {
                    this.$store.state.toast.showToast("Personality remounted", 4, true)

                    return res.data

                }
                else{
                    console.log("failed remount_personality")
                }
            } catch (error) {
                console.log(error.message, 'remount_personality - settings')
                return
            }

        },      
        async unmountPersonality(pers) {
            if (!pers) { return }

            const res = await this.unmount_personality(pers.personality || pers)
            if (res.status) {
                this.$store.state.config.personalities = res.personalities
                this.$store.state.toast.showToast("Personality unmounted", 4, true)

                //pers.isMounted = false
                this.$store.dispatch('refreshMountedPersonalities');
                // Select some other personality
                const lastPers = this.$store.state.mountedPersArr[this.$store.state.mountedPersArr.length - 1]

                // const res2 = await this.select_personality(lastPers.personality)
                const res2 = await this.select_personality(pers.personality)
                if (res2.status) {
                    this.$store.state.toast.showToast("Selected personality:\n" + lastPers.name, 4, true)
                }


            } else {
                this.$store.state.toast.showToast("Could not unmount personality\nError: " + res.error, 4, false)
            }

        },

        async unmount_personality(pers) {
            if (!pers) { return { 'status': false, 'error': 'no personality - unmount_personality' } }

            const obj = {
                client_id: this.$store.state.client_id,
                language: pers.language,
                category: pers.category,
                folder: pers.folder
            }


            try {
                const res = await axios.post('/unmount_personality', obj);

                if (res) {
                    return res.data

                }
            } catch (error) {
                console.log(error.message, 'unmount_personality - settings')
                return
            }

        },
        
        handleShortcut(event) {
            if (event.ctrlKey && event.key === 'd') {
                event.preventDefault();
                event.stopPropagation();                
                this.createNewDiscussion();
            } 
        },

        toggleMenu() {
            this.isMenuVisible = !this.isMenuVisible;
            if (this.isMenuVisible){
                this.isinfosMenuVisible=false;
            }
            nextTick(() => {
                feather.replace()

            })            
        },  
        showMenu() {
            this.isMenuVisible = true;
            nextTick(() => {
                feather.replace()

            })            
        },              
        hideMenu() {
            this.isMenuVisible = false;
            nextTick(() => {
                feather.replace()

            })            
        },    
        adjustMenuPosition() {
            const menu = this.$refs.languageMenu;
            if(menu){
                const rect = menu.getBoundingClientRect();
                const windowWidth = window.innerWidth;

                if (rect.right > windowWidth) {
                    menu.style.left = 'auto';
                    menu.style.right = '0';
                } else {
                    menu.style.left = '0';
                    menu.style.right = 'auto';
                }
            }
        },
        restartProgram(event) {
            event.preventDefault();
            this.$store.state.api_post_req('restart_program', this.$store.state.client_id)
            this.rebooting_audio.play()
            this.$store.state.toast.showToast("Rebooting the app. Please wait...", 410, false)
            //this.$store.state.toast.showToast("Rebooting the app. Please wait...", 50, true);
            setTimeout(()=>{
                window.close();
            },2000)
        },  
        applyConfiguration() {
            this.isLoading = true;
            axios.post('/apply_settings', {"client_id":this.$store.state.client_id, "config":this.$store.state.config}, {headers: this.posts_headers}).then((res) => {
                this.isLoading = false;
                if (res.data.status) {

                    this.$store.state.toast.showToast("Configuration changed successfully.", 4, true)
                    this.settingsChanged = false
                    //this.save_configuration()
                } else {

                    this.$store.state.toast.showToast("Configuration change failed.", 4, false)

                }
                nextTick(() => {
                    feather.replace()

                })
            })
        },

        getRandomEdgePosition() {
            const edge = Math.floor(Math.random() * 4);
            switch (edge) {
                case 0: return 0; // Top edge
                case 1: return 100; // Right edge
                case 2: return Math.random() * 100; // Bottom or left edge
                case 3: return Math.random() * 100; // Bottom or left edge
            }
        },
        extractTitle(prompt) {
            const titleMatch = prompt.match(/@<(.*?)>@/);
            return titleMatch ? titleMatch[1] : null;
        },
        getPromptContent(prompt) {
            // Remove the title tag if it exists and return the remaining content
            return prompt.replace(/@<.*?>@/, '').trim();
        },        
        handlePromptSelection(prompt) {
            this.selectedPrompt = prompt;
            const title = this.extractTitle(prompt)
            if (title){
                this.previewPrompt = this.getPromptContent(prompt); // Initialize preview
            }
            else{
                this.previewPrompt = prompt; // Initialize preview

            }
            this.placeholders = this.extractPlaceholders(prompt);
            
            if (this.placeholders.length > 0) {
                this.showPlaceholderModal = true;
                this.placeholderValues = {};
            } else {
                this.setPromptInChatbox(prompt);
            }
        },

        updatePreview() {
            let preview = this.selectedPrompt;
            this.parsedPlaceholders.forEach((placeholder, index) => {
                const value = this.placeholderValues[index];
                // Replace all occurrences of the same placeholder
                const regex = new RegExp(this.escapeRegExp(placeholder.fullText), 'g');
                preview = preview.replace(regex, value || placeholder.fullText);
            });
            this.previewPrompt = preview;
        },
        escapeRegExp(string) {
            return string.replace(/[.*+?^${}()|[\]\\]/g, '\\$&');
        },
        cancelPlaceholders() {
            this.showPlaceholderModal = false;
            this.placeholders = [];
            this.placeholderValues = {};
            this.previewPrompt = '';
        },

        // When applying placeholders, make sure to update all instances
        applyPlaceholders() {
            let finalPrompt = this.selectedPrompt;
            this.parsedPlaceholders.forEach((placeholder, index) => {
                const value = this.placeholderValues[index];
                if (value) {
                    const regex = new RegExp(this.escapeRegExp(placeholder.fullText), 'g');
                    finalPrompt = finalPrompt.replace(regex, value);
                }
            });
            // Apply the final prompt and close modal
            this.finalPrompt = finalPrompt;
            this.showPlaceholderModal = false;

            this.setPromptInChatbox(this.getPromptContent(this.previewPrompt));
        },

        extractPlaceholders(prompt) {
            const placeholderRegex = /\[(.*?)\]/g;
            return [...prompt.matchAll(placeholderRegex)].map(match => match[0]);
        },

        setPromptInChatbox(prompt) {
            this.$refs.chatBox.message = prompt;
        },
        extractHtml() {
            if (this.discussionArr.length > 0) {
                const lastMessage = this.discussionArr[this.discussionArr.length - 1].content;
                const startTag = '```html';
                const endTag = '```';
                let startIndex = lastMessage.indexOf(startTag);
                if (startIndex === -1) {
                    this.lastMessageHtml = this.defaultMessageHtml;
                    this.renderIsolatedContent();
                    return this.defaultMessageHtml;
                }
                
                startIndex += startTag.length;
                let endIndex = lastMessage.indexOf(endTag, startIndex);
                
                if (endIndex === -1) {
                    this.lastMessageHtml = lastMessage.slice(startIndex).trim();
                } else {
                    this.lastMessageHtml = lastMessage.slice(startIndex, endIndex).trim();
                }
            } else {
                this.lastMessageHtml = this.defaultMessageHtml;
            }
            this.renderIsolatedContent()
        },
        renderIsolatedContent() {
            const iframe = document.createElement('iframe');
            iframe.style.border = 'none';
            iframe.style.width = '100%';
            iframe.style.height = '100%'; // Adjust as needed
            if (this.$refs.isolatedContent){
                this.$refs.isolatedContent.innerHTML = '';
                this.$refs.isolatedContent.appendChild(iframe);
                
                const iframeDoc = iframe.contentDocument || iframe.contentWindow.document;
                iframeDoc.open();
                iframeDoc.write(`
                    ${this.lastMessageHtml}
                `);
                iframeDoc.close();
            }

        },        
        async triggerRobotAction(){
            this.rightPanelCollapsed = !this.rightPanelCollapsed;
            if(!this.rightPanelCollapsed){
                this.$store.commit('setleftPanelCollapsed', false); // Assuming you have a mutation to set the view mode
                this.$nextTick(() => {
                this.extractHtml()
                });

            }
        }, 
        add_webpage(){
            this.$store.state.web_url_input_box.showPanel();
        },
        addWebpage(){

            axios.post('/add_webpage', {"client_id":this.client_id, "url": this.$store.state.web_url_input_box.inputText}, {headers: this.posts_headers}).then(response => {
                if (response && response.status){
                    this.recoverFiles()
                }
            });

        },
        show_progress(data){
            this.progress_visibility_val = true;
        },
        hide_progress(data){
            this.progress_visibility_val = false;
        },
        update_progress(data){
            this.progress_value = data.value;
        },
        onSettingsBinding() {
            try {
                this.isLoading = true
                axios.get('/get_active_binding_settings').then(res => {
                    this.isLoading = false
                    if (res) {
                        if (res.data && Object.keys(res.data).length > 0) {

                            // open form
                            const bindingEntry = this.$store.state.bindingsZoo.find(item=>item.name == this.state.config.binding_name)
                            this.$store.state.universalForm.showForm(res.data, "Binding settings - " + bindingEntry.binding.name, "Save changes", "Cancel").then(res => {
                                // send new data
                                try {
                                    axios.post('/set_active_binding_settings',
                                    {client_id:this.$store.state.client_id, "settings":res}).then(response => {

                                            if (response && response.data) {
                                                this.$store.state.toast.showToast("Binding settings updated successfully!", 4, true)

                                            } else {
                                                this.$store.state.toast.showToast("Did not get binding settings responses.\n" + response, 4, false)
                                                this.isLoading = false
                                            }


                                        })
                                } catch (error) {
                                    this.$store.state.toast.showToast("Did not get binding settings responses.\n Endpoint error: " + error.message, 4, false)
                                    this.isLoading = false
                                }



                            })
                        } else {
                            this.$store.state.toast.showToast("Binding has no settings", 4, false)
                            this.isLoading = false
                        }

                    }
                })

            } catch (error) {
                this.isLoading = false
                this.$store.state.toast.showToast("Could not open binding settings. Endpoint error: " + error.message, 4, false)
            }
        },        
        showDatabaseSelector() {
            this.database_selectorDialogVisible = true;
        },   
             
        async ondatabase_selectorDialogRemoved(choice){
            console.log("Deleted:",choice)
        },
        async ondatabase_selectorDialogSelected(choice){
            console.log("Selected:",choice)
        },
        onclosedatabase_selectorDialog(){this.database_selectorDialogVisible=false;},
        async onvalidatedatabase_selectorChoice(choice){
            this.database_selectorDialogVisible=false;
            const data = {
                "client_id": this.client_id,
                "name": typeof choice === 'string' ? choice : choice.name
            };
            const res = await axios.post("/select_database", data, {headers: this.posts_headers});
            if(res.status){
                this.$store.state.config = await axios.post("/get_config",{"client_id":this.client_id});
                let dbs = await axios.get("/list_databases")["data"];
                this.$store.state.databases = dbs
                location.reload();
            }
        
        },
        async addDiscussion2SkillsLibrary(){
            let result = await axios.post("/add_discussion_to_skills_library", {
                        client_id: this.client_id
                    }, {headers: this.posts_headers});
            if(result.status){
                console.log("done")
            }
        },
        async toggleSkillsLib(){
            this.$store.state.config.activate_skills_lib =! this.$store.state.config.activate_skills_lib;
            await this.applyConfiguration();
        },
        async showSkillsLib(){
            this.$refs.skills_lib.showSkillsLibrary()
        },
        
        async applyConfiguration() {
            this.loading = true;
            const res = await axios.post('/apply_settings', {"client_id":this.$store.state.client_id, "config":this.$store.state.config})
            this.loading = false;
            //console.log('apply-res',res)
            if (res.data.status) {
                
                this.$store.state.toast.showToast("Configuration changed successfully.", 4, true)
                //this.save_configuration()
            } else {
                this.$store.state.toast.showToast("Configuration change failed.", 4, false)
            }
            nextTick(() => {
                feather.replace()
            })
        },
        save_configuration() {
            this.showConfirmation = false
            axios.post('/save_settings', {})
                .then((res) => {
                    if (res) {
                        if (res.status) {
                            this.$store.state.toast.showToast("Settings saved!",4,true)
                        }
                        else
                            this.$store.state.messageBox.showMessage("Error: Couldn't save settings!")
                        return res.data;
                    }
                })
                .catch(error => {
                    this.$store.state.messageBox.showMessage("Couldn't save settings!")
                    return { 'status': false }
                });

        },        
        showToastMessage(text, duration, isok){
            this.$store.state.toast.showToast(text, duration, isok)
        },        
        toggleDropdown() {
            this.isOpen = !this.isOpen;
        },
        importChatGPT() {
            // handle ChatGPT import
        },  
        async api_get_req(endpoint) {
            try {
                const res = await axios.get("/" + endpoint);

                if (res) {

                    return res.data

                }
            } catch (error) {
                console.log(error.message, 'api_get_req')
                return
            }
        },
        async list_discussions() {
            try {
                const res = await axios.get('/list_discussions')

                if (res) {

                    this.createDiscussionList(res.data)
                    return res.data
                }
            } catch (error) {
                console.log("Error: Could not list discussions", error.message)
                return []
            }
        },
        load_discussion(id, next) {
            if (id) {
                this.loading = true
                this.discussionArr=[]
                this.setDiscussionLoading(id, this.loading)


                socket.on('discussion', (data)=>{
                    this.loading = false
                    this.setDiscussionLoading(id, this.loading)
                    if (data) {
                        // Filter out the user and bot entries
                        this.discussionArr = data.filter((item) => 
                                                                item.message_type == this.msgTypes.MSG_TYPE_CONTENT ||
                                                                item.message_type == this.msgTypes.MSG_TYPE_CONTENT_INVISIBLE_TO_AI
                                                        )
                        this.discussionArr.forEach((item) => {
                            item.status_message = "Done";
                        });                          
                        
                        if(next){
                            next()
                        }
                    }
                    
                    socket.off('discussion')
                    this.extractHtml()
                })

                socket.emit('load_discussion',{"id":id});

            }
        },
        recoverFiles(){
            axios.post('/get_discussion_files_list', {"client_id":this.$store.state.client_id}).then(res=>{
                this.$refs.chatBox.filesList = res.data.files;
                this.$refs.chatBox.isFileSentList= res.data.files.map(file => {
                    return true;
                });
            })
        },
        new_discussion(title) {
            try {
                this.loading = true
                socket.on('discussion_created',(data)=>{
                    socket.off('discussion_created')
                    this.list_discussions().then(()=>{
                        const index = this.discussionsList.findIndex((x) => x.id == data.id)
                        const discussionItem = this.discussionsList[index]
                        this.selectDiscussion(discussionItem)
                        this.load_discussion(data.id,()=>{
                            this.loading = false
                            this.recoverFiles();
                            nextTick(() => {
                                const selectedDisElement = document.getElementById('dis-' + data.id)
                                this.scrollToElement(selectedDisElement)
                                console.log("Scrolling tp "+selectedDisElement)
                            })
                        })
                    });
                });
                socket.emit('new_discussion', {title:title});
            } catch (error) {
                console.log("Error: Could not create new discussion", error.message)
                return {}
            }
        },
        async delete_discussion(id) {
            try {
                if (id) {
                    this.loading = true
                    this.setDiscussionLoading(id, this.loading)
                    await axios.post('/delete_discussion', {
                        client_id: this.client_id,
                        id: id
                    }, {headers: this.posts_headers})
                    this.loading = false
                    this.setDiscussionLoading(id, this.loading)
                }
            } catch (error) {
                console.log("Error: Could not delete discussion", error.message)
                this.loading = false
                this.setDiscussionLoading(id, this.loading)
            }
        },
        async edit_title(id, new_title) {
            try {
                if (id) {
                    this.loading = true
                    this.setDiscussionLoading(id, this.loading)
                    const res = await axios.post('/edit_title', {
                        client_id: this.client_id,
                        id: id,
                        title: new_title
                    }, {headers: this.posts_headers})
                    this.loading = false
                    this.setDiscussionLoading(id, this.loading)
                    if (res.status == 200) {
                        const index = this.discussionsList.findIndex((x) => x.id == id)
                        const discussionItem = this.discussionsList[index]
                        discussionItem.title = new_title
                        this.tempList = this.discussionsList
                    }
                }
            } catch (error) {
                console.log("Error: Could not edit title", error.message)
                this.loading = false
                this.setDiscussionLoading(id, this.loading)
            }
        },
        async make_title(id) {
            try {
                if (id) {
                    this.loading = true
                    this.setDiscussionLoading(id, this.loading)
                    const res = await axios.post('/make_title', {
                        client_id: this.client_id,
                        id: id,
                    }, {headers: this.posts_headers})

                    this.loading = false
                    this.setDiscussionLoading(id, this.loading)
                    if (res.status == 200) {
                        const index = this.discussionsList.findIndex((x) => x.id == id)
                        const discussionItem = this.discussionsList[index]
                        discussionItem.title = res.data.title
                        
                        this.tempList = this.discussionsList
                    }
                }
            } catch (error) {
                console.log("Error: Could not edit title", error.message)
                this.loading = false
                this.setDiscussionLoading(id, this.loading)
            }
        },        
        async delete_message(id) {
            try {
                const res = await axios.post('/delete_message', { client_id: this.client_id, id: id }, {headers: this.posts_headers})

                if (res) {
                    return res.data
                }
            } catch (error) {
                console.log("Error: Could delete message", error.message)
                return {}
            }
        },

        
        async stop_gen() {
            try {
                if (this.discussionArr.length>0)
                {
                    const messageItem = this.discussionArr[this.discussionArr.length-1]            
                    messageItem.status_message = "Generation canceled"
                }
                socket.emit('cancel_generation');
                //const res = await axios.get('/stop_gen')

                if (res) {
                    return res.data
                }
            } catch (error) {
                console.log("Error: Could not stop generating", error.message)
                return {}
            }
        },
        async message_rank_up(id) {
            try {
                const res = await axios.post('/message_rank_up', { client_id: this.client_id, id: id }, {headers: this.posts_headers})

                if (res) {
                    return res.data
                }
            } catch (error) {
                console.log("Error: Could not rank up message", error.message)
                return {}
            }
        },
        async message_rank_down(id) {
            try {
                const res = await axios.post('/message_rank_down', { client_id: this.client_id, id: id } , {headers: this.posts_headers})

                if (res) {
                    return res.data
                }
            } catch (error) {
                console.log("Error: Could not rank down message", error.message)
                return {}
            }
        },
        async edit_message(id, message, audio_url) {
            try {
                const res = await axios.post('/edit_message', {
                                                            client_id: this.client_id, 
                                                            id: id, 
                                                            message: message,
                                                            metadata: [{audio_url:audio_url}]
                                                    }, {headers: this.posts_headers}
                )

                if (res) {
                    return res.data
                }
            } catch (error) {
                console.log("Error: Could not update message", error.message)
                return {}
            }
        },
        async export_multiple_discussions(discussionIdArr, export_format) {
            try {
                if (discussionIdArr.length > 0) {
                    const res = await axios.post('/export_multiple_discussions', {
                        client_id:this.$store.state.client_id,
                        discussion_ids: discussionIdArr,
                        export_format: export_format
                    }, {headers: this.posts_headers})

                    if (res) {
                        return res.data
                    }
                }

            } catch (error) {
                console.log("Error: Could not export multiple discussions", error.message)
                return {}
            }
        },
        async import_multiple_discussions(jArray_) {
            try {
                if (jArray_.length > 0) {
                    const res = await axios.post('/import_multiple_discussions', {
                        client_id: this.$store.state.client_id,
                        jArray: jArray_
                    }, {headers: this.posts_headers})

                    if (res) {
                        return res.data
                    }
                }

            } catch (error) {
                console.log("Error: Could not import multiple discussions", error.message)
                return
            }
        },
        handleSearch() {
            if (!this.filterTitle.trim()) return
            
            this.isSearching = true
            // Add debounce to prevent too frequent searches
            clearTimeout(this.searchTimeout)
            this.searchTimeout = setTimeout(() => {
                this.filterDiscussions()
                this.isSearching = false
            }, 300)
        },
        
        clearSearch() {
            this.filterTitle = ''
            this.searchResults = []
        },        
        filterDiscussions() {
            // Search bar in for filtering discussions by title (serch)

            if (!this.filterInProgress) {
                this.filterInProgress = true
                setTimeout(() => {
                    if (this.filterTitle) {
                        this.discussionsList = this.tempList.filter((item) => item.title && item.title.includes(this.filterTitle))

                    } else {
                        this.discussionsList = this.tempList
                    }
                    this.filterInProgress = false
                }, 100)
            }
        },
        async selectDiscussion(item) {
            if(this.isGenerating){
                this.$store.state.toast.showToast("You are currently generating a text. Please wait for text generation to finish or stop it before trying to select another discussion", 4, false)
                return;
            }

            if (item) {
                // When discussion is selected it loads the discussion array
                if (this.currentDiscussion===undefined) {
                    this.currentDiscussion = item

                    this.setPageTitle(item)

                    localStorage.setItem('selected_discussion', this.currentDiscussion.id)
                    const discussion_id = localStorage.getItem('selected_discussion')

                    this.load_discussion(item.id, ()=>{
                        if (this.discussionArr.length > 1) {
                        if (this.currentDiscussion.title === '' || this.currentDiscussion.title === null) {
                            this.changeTitleUsingUserMSG(this.currentDiscussion.id, this.discussionArr[1].content)
                        }
                        this.recoverFiles()
                    }
                    })

                }
                else{
                    if (this.currentDiscussion.id != item.id) {
                        this.currentDiscussion = item
                        this.setPageTitle(item)

                        localStorage.setItem('selected_discussion', this.currentDiscussion.id)
                        this.load_discussion(item.id, ()=>{
                            if (this.discussionArr.length > 1) {
                                if (this.currentDiscussion.title === '' || this.currentDiscussion.title === null) {
                                    this.changeTitleUsingUserMSG(this.currentDiscussion.id, this.discussionArr[1].content)
                                }
                            }
                            this.recoverFiles()
                        });

                    }
                }

                nextTick(() => {


                    const discussionitem = document.getElementById('dis-' + this.currentDiscussion.id)

                    //this.scrollToElement(discussionitem)

                    this.scrollToElementInContainer(discussionitem, 'leftPanel')
                    const msgList = document.getElementById('messages-list')

                    this.scrollBottom(msgList)

                })
            }
        },

        scrollToElement(el) {

            if (el) {
                el.scrollIntoView({ behavior: 'smooth', block: 'start', inline: 'nearest' })
            } else {
                console.log("Error: scrollToElement")
            }
        },
        scrollToElementInContainer(el, containerId) {
            try{
                const topPos = el.offsetTop; //+ el.clientHeight
                const container = document.getElementById(containerId)
                // console.log(el.offsetTop , el.clientHeight, container.clientHeight)


                container.scrollTo(
                    {
                        top: topPos,
                        behavior: "smooth",
                    }
                )

            }
            catch{
                console.log("error")
            }

        },
        scrollBottom(el) {

            if (el) {
                console.log("Scrolling to bottom")
                el.scrollTo(
                    {
                        top: el.scrollHeight,
                        behavior: "smooth",
                    }
                )
            } else {
                console.log("Error: scrollBottom")
            }

        },

        scrollTop(el) {

            if (el) {
                el.scrollTo(
                    {
                        top: 0,
                        behavior: "smooth",
                    }
                )
            } else {
                console.log("Error: scrollTop")
            }

        },
        createUserMsg(msgObj) {

            let usrMessage = {
                content: msgObj.message,
                id: msgObj.id,
                rank: 0,
                sender: msgObj.user,
                created_at: msgObj.created_at,
                steps: [],
                html_js_s: [],
                status_message: "Warming up"

            }
            this.discussionArr.push(usrMessage)
            nextTick(() => {
                const msgList = document.getElementById('messages-list')
                if(msgList.length>0){
                    this.scrollBottom(msgList)
                }

            })
        },
        updateLastUserMsg(msgObj) {

            // const lastMsg = this.discussionArr[this.discussionArr.length - 1]
            // lastMsg.content = msgObj.message
            // lastMsg.id = msgObj.user_id
            // // lastMsg.parent=msgObj.parent
            // lastMsg.rank = msgObj.rank
            // lastMsg.sender = msgObj.user
            // // lastMsg.type=msgObj.type
            const index = this.discussionArr.indexOf(item => item.id = msgObj.user_id)
            const newMessage ={
                binding: msgObj.binding,
                content: msgObj.message,
                created_at: msgObj.created_at,
                type: msgObj.type,
                finished_generating_at: msgObj.finished_generating_at,
                id: msgObj.user_id,
                model: msgObj.model,
                personality: msgObj.personality,
                sender: msgObj.user,
                steps:[]
            }
            
            
            if (index !== -1) {
                this.discussionArr[index] = newMessage;
            }

        },
        async socketIOConnected() {
            console.log("socketIOConnected")
 
        },
        socketIODisconnected() {
            console.log("socketIOConnected")
            this.currentDiscussion=null
            this.$store.dispatch('refreshModels');
            this.$store.state.isConnected=false;
            return true
        },
        new_message(msgObj) {
            if(msgObj.sender_type==this.SENDER_TYPES_AI){
                this.isGenerating = true
            }
            
            let responseMessage = {
                sender:                 msgObj.sender,
                message_type:           msgObj.message_type,
                sender_type:            msgObj.sender_type,
                content:                msgObj.content,//"✍ please stand by ...",
                id:                     msgObj.id,
                discussion_id:          msgObj.discussion_id,
                parent_id:              msgObj.parent_id,

                binding:                msgObj.binding,
                model:                  msgObj.model,
                personality:            msgObj.personality,

                created_at:             msgObj.created_at,
                finished_generating_at: msgObj.finished_generating_at,
                rank:                   0,

                ui:                     msgObj.ui,

                steps                   : [],
                parameters              : msgObj.parameters,
                metadata                : msgObj.metadata,

                open                    : msgObj.open
            }
            
            responseMessage.status_message = "Warming up"
            this.discussionArr.push(responseMessage)
            nextTick(() => {
                 const msgList = document.getElementById('messages-list')

                 this.scrollBottom(msgList)
            });

            if (this.currentDiscussion.title === '' || this.currentDiscussion.title === null) {
                this.changeTitleUsingUserMSG(this.currentDiscussion.id, msgObj.message)
            }
        },
        async talk(pers){
            this.isGenerating = true;
            this.setDiscussionLoading(this.currentDiscussion.id, this.isGenerating);
            let res = await axios.get('/get_generation_status', {})
            if (res) {
                if (!res.data.status) {
                    const id = this.$store.state.config.personalities.findIndex(item => item === pers.full_path)
                    const obj = {
                    client_id:this.$store.state.client_id,
                    id: id
                    }
                    res = await axios.post('/select_personality', obj);
                    socket.emit('generate_msg_from', { id: -1 });
                }
                else {
                    console.log("Already generating");
                }
            }
        },
        createEmptyUserMessage(message){
            socket.emit('create_empty_message', {"type":0,"message":message}); // 0 for user and 1 for AI
        },
        createEmptyAIMessage(){
            socket.emit('create_empty_message', {"type":1}); // 0 for user and 1 for AI
        },
        sendMsg(msg,type) {
            // Sends message to binding
            if (!msg) {
                this.$store.state.toast.showToast("Message contains no content!", 4, false)
                return
            }
            this.isGenerating = true;
            this.setDiscussionLoading(this.currentDiscussion.id, this.isGenerating);
            axios.get('/get_generation_status', {}).then((res) => {
                if (res) {
                    if (!res.data.status) {
                        if(type=="internet"){
                            socket.emit('generate_msg_with_internet', { prompt: msg });
                        }
                        else{
                            socket.emit('generate_msg', { prompt: msg });
                        }

                        // Create new User message
                        // Temp data
                        let lastmsgid =0
                        if(this.discussionArr.length>0){
                            lastmsgid= Number(this.discussionArr[this.discussionArr.length - 1].id) + 1
                        }
                        
                        
                        let usrMessage = {
                            message: msg,
                            id: lastmsgid,
                            rank: 0,
                            user: this.$store.state.config.user_name,
                            created_at: new Date().toLocaleString(),


                            sender:                 this.$store.state.config.user_name,
                            message_type:           this.operationTypes.MSG_TYPE_CONTENT,
                            sender_type:            this.senderTypes.SENDER_TYPES_USER,
                            content:                msg,
                            id:                     lastmsgid,
                            discussion_id:          this.discussion_id,
                            parent_id:              lastmsgid,

                            binding:                "",
                            model:                  "",
                            personality:            "",

                            created_at:             new Date().toLocaleString(),
                            finished_generating_at: new Date().toLocaleString(),
                            rank:                   0,

                            steps:                  [],
                            parameters:             null,
                            metadata:               [],
                            ui:                     null

                        };
                        this.createUserMsg(usrMessage);

                    }
                    else {
                        console.log("Already generating");
                    }
                }
            }).catch((error) => {
                console.log("Error: Could not get generation status", error);
            });
        },
        sendCmd(cmd){
            this.isGenerating = true;            
            socket.emit('execute_command', { command: cmd, parameters: [] });            
        },
        notify(notif){
            self.isGenerating = false
            this.setDiscussionLoading(this.currentDiscussion.id, this.isGenerating);
            nextTick(() => {
                const msgList = document.getElementById('messages-list')
                this.scrollBottom(msgList)
            })
            if(notif.display_type==0){
                this.$store.state.toast.showToast(notif.content, notif.duration, notif.notification_type)
            }
            else if(notif.display_type==1){
                this.$store.state.messageBox.showMessage(notif.content)
            }
            else if(notif.display_type==2){
                this.$store.state.messageBox.hideMessage()
                this.$store.state.yesNoDialog.askQuestion(notif.content, 'Yes', 'No').then(yesRes => {
                    socket.emit("yesNoRes",{yesRes:yesRes})
                })
            }
            else if(notif.display_type==3){
                this.$store.state.messageBox.showBlockingMessage(notif.content)
            }
            else if(notif.display_type==4){
                this.$store.state.messageBox.hideMessage()
            }
            
            this.chime.play()
        },
        update_message(msgObj) {
            // Streams response message content from binding
            this.discussion_id = msgObj.discussion_id
            this.setDiscussionLoading(this.discussion_id, true);

            if (this.currentDiscussion.id == this.discussion_id) {
                //this.isGenerating = true;
                const index = this.discussionArr.findIndex((x) => x.id == msgObj.id)
                const messageItem = this.discussionArr[index]
                if (
                    messageItem && (msgObj.operation_type==this.operationTypes.MSG_OPERATION_TYPE_SET_CONTENT ||
                    msgObj.operation_type==this.operationTypes.MSG_OPERATION_TYPE_SET_CONTENT_INVISIBLE_TO_AI)
                ) {
                    this.isGenerating = true;
                    messageItem.content = msgObj.content
                    messageItem.created_at = msgObj.created_at
                    messageItem.started_generating_at = msgObj.started_generating_at
                    messageItem.nb_tokens = msgObj.nb_tokens
                    messageItem.finished_generating_at = msgObj.finished_generating_at
                    this.extractHtml()
                }
                else if(messageItem && msgObj.operation_type==this.operationTypes.MSG_OPERATION_TYPE_ADD_CHUNK){
                    this.isGenerating = true;
                    messageItem.content += msgObj.content
                    messageItem.created_at              = msgObj.created_at
                    messageItem.started_generating_at   = msgObj.started_generating_at
                    messageItem.nb_tokens               = msgObj.nb_tokens
                    messageItem.finished_generating_at  = msgObj.finished_generating_at
                    this.extractHtml()
                } else if (msgObj.operation_type == this.operationTypes.MSG_OPERATION_TYPE_STEP || msgObj.operation_type == this.operationTypes.MSG_OPERATION_TYPE_STEP_START || msgObj.operation_type == this.operationTypes.MSG_OPERATION_TYPE_STEP_END_SUCCESS || msgObj.operation_type == this.operationTypes.MSG_OPERATION_TYPE_STEP_END_FAILURE){
                    if (Array.isArray(msgObj.steps)) {
                        console.log("Received steps:", msgObj.steps)
                        messageItem.status_message = msgObj.steps[msgObj.steps.length - 1]["text"]
                        messageItem.steps = msgObj.steps;
                    } else {
                        console.error("Invalid steps data:", msgObj.steps);
                    }

                } else if (msgObj.operation_type == this.operationTypes.MSG_OPERATION_TYPE_JSON_INFOS) {
                    if (typeof msgObj.metadata === 'string') {
                        try {
                            messageItem.metadata = JSON.parse(msgObj.metadata);
                        } catch (error) {
                            console.error("Error parsing metadata string:", error);
                            messageItem.metadata = { raw: msgObj.metadata }; // Fallback: store as raw string in an object
                        }
                    } else if (Array.isArray(msgObj.metadata) || typeof msgObj.metadata === 'object') {
                        messageItem.metadata = msgObj.metadata; // Already an array or object, assign directly
                    } else {
                        messageItem.metadata = { value: msgObj.metadata }; // For any other type, wrap in an object
                    }
                } else if (msgObj.operation_type == this.operationTypes.MSG_OPERATION_TYPE_UI) {
                    messageItem.ui = msgObj.ui
                } else if (msgObj.operation_type == this.operationTypes.MSG_OPERATION_TYPE_EXCEPTION) {
                    this.$store.state.toast.showToast(msgObj.content, 5, false)
                }
                // // Disables as per request
                // nextTick(() => {
                //     const msgList = document.getElementById('messages-list')
                //     this.scrollBottom(msgList)
                // })
            }
            // Force an immediate UI update
            this.$nextTick(() => {
                // UI updates are rendered here
                feather.replace()
            });

        },
        async changeTitleUsingUserMSG(id, msg) {
            // If discussion is untitled or title is null then it sets the title to first user message.

            const index = this.discussionsList.findIndex((x) => x.id == id)
            const discussionItem = this.discussionsList[index]
            if (msg) {
                discussionItem.title = msg
                this.tempList = this.discussionsList
                await this.edit_title(id, msg)
            }

        },
        async createNewDiscussion() {
            // Creates new discussion on binding,
            // gets new discussion list, selects
            // newly created discussion,
            // scrolls to the discussion
            this.new_discussion(null)
        },
        loadLastUsedDiscussion() {
            // Checks local storage for last selected discussion
            const id = localStorage.getItem('selected_discussion')
            if (id) {
                const index = this.discussionsList.findIndex((x) => x.id == id)
                const discussionItem = this.discussionsList[index]
                if (discussionItem) {
                    this.selectDiscussion(discussionItem)
                }
            }
        },        
        onCopyPersonalityName(personality) {
            this.$store.state.toast.showToast("Copied name to clipboard!", 4, true)
            navigator.clipboard.writeText(personality.name);
        },
        async deleteDiscussion(id) {
            // Deletes discussion from binding and frontend

            await this.delete_discussion(id)
            if (this.currentDiscussion.id == id) {
                this.currentDiscussion = {}
                this.discussionArr = []
                this.setPageTitle()
            }
            this.discussionsList.splice(this.discussionsList.findIndex(item => item.id == id), 1)

            this.createDiscussionList(this.discussionsList)
        },
        async deleteDiscussionMulti() {
            // Delete selected discussions

            const deleteList = this.selectedDiscussions

            for (let i = 0; i < deleteList.length; i++) {
                const discussionItem = deleteList[i]
                //discussionItem.loading = true
                await this.delete_discussion(discussionItem.id)
                if (this.currentDiscussion.id == discussionItem.id) {
                    this.currentDiscussion = {}
                    this.discussionArr = []
                    this.setPageTitle()
                }
                this.discussionsList.splice(this.discussionsList.findIndex(item => item.id == discussionItem.id), 1)
            }
            this.tempList = this.discussionsList
            this.isCheckbox = false
            this.$store.state.toast.showToast("Removed (" + deleteList.length + ") items", 4, true)
            this.showConfirmation = false
        },
        async deleteMessage(msgId) {

            await this.delete_message(msgId).then(() => {

                this.discussionArr.splice(this.discussionArr.findIndex(item => item.id == msgId), 1)

            }).catch(() => {
                this.$store.state.toast.showToast("Could not remove message", 4, false)
                console.log("Error: Could not delete message")
            })

        },
        async openFolder(id){
            const json = JSON.stringify({ 'client_id': this.$store.state.client_id, 'discussion_id': id.id })   
            await axios.post(`/open_discussion_folder`, json, {
                method: 'POST',
                headers: { 'Content-Type': 'application/json' },
            })
        },
        async openCustomFunctionCallsFolder(id){
            const json = JSON.stringify({ 'client_id': this.$store.state.client_id, 'discussion_id': id.id })   
            await axios.post(`/open_custom_function_calls_folder`, json, {
                method: 'POST',
                headers: { 'Content-Type': 'application/json' },
            })
        },
        async editTitle(newTitleObj) {

            const index = this.discussionsList.findIndex((x) => x.id == newTitleObj.id)
            const discussionItem = this.discussionsList[index]
            discussionItem.title = newTitleObj.title
            discussionItem.loading = true
            await this.edit_title(newTitleObj.id, newTitleObj.title)
            discussionItem.loading = false
        },
        async makeTitle(editTitleObj) {
            const index = this.discussionsList.findIndex((x) => x.id == editTitleObj.id)
            await this.make_title(editTitleObj.id)
        },

        checkUncheckDiscussion(event, id) {
            // If checked = true and item is not in array then add item to list
            const index = this.discussionsList.findIndex((x) => x.id == id)
            const discussionItem = this.discussionsList[index]
            discussionItem.checkBoxValue = event.target.checked
            this.tempList = this.discussionsList
        },
        selectAllDiscussions() {

            // Check if there is one discussion not selected
            this.isSelectAll = !this.tempList.filter((item) => item.checkBoxValue == false).length > 0
            // Selects or deselects all discussions
            for (let i = 0; i < this.tempList.length; i++) {
                this.tempList[i].checkBoxValue = !this.isSelectAll
            }

            this.tempList = this.discussionsList
            this.isSelectAll = !this.isSelectAll
        },
        createDiscussionList(disList) {
            // This creates a discussion list for UI with additional properties
            if (disList) {
                const newDisList = disList.map((item) => {

                    const newItem = {
                        id: item.id,
                        title: item.title,
                        selected: false,
                        loading: false,
                        checkBoxValue: false
                    }
                    return newItem

                }).sort(function (a, b) {
                    return b.id - a.id
                })

                this.discussionsList = newDisList
                this.tempList = newDisList
            }
        },
        setDiscussionLoading(id, loading) {
            try{
                const index = this.discussionsList.findIndex((x) => x.id == id)
                const discussionItem = this.discussionsList[index]
                discussionItem.loading = loading
            }
            catch{
                console.log("Error setting discussion loading")
            }
        },
        setPageTitle(item) {
            // item is either title:String or {id:Number, title:String}
            if (item) {
                if (item.id) {
                    const realTitle = item.title ? item.title === "untitled" ? "New discussion" : item.title : "New discussion"
                    document.title = 'L🌟LLMS WebUI - ' + realTitle
                } else {
                    const title = item || "Welcome"
                    document.title = 'L🌟LLMS WebUI - ' + title
                }
            } else {
                const title = item || "Welcome"
                document.title = 'L🌟LLMS WebUI - ' + title
            }

        },
        async rankUpMessage(msgId) {
            await this.message_rank_up(msgId).then((res) => {

                const message = this.discussionArr[this.discussionArr.findIndex(item => item.id == msgId)]
                message.rank = res.new_rank
            }).catch(() => {
                this.$store.state.toast.showToast("Could not rank up message", 4, false)
                console.log("Error: Could not rank up message")
            })

        },
        async rankDownMessage(msgId) {
            await this.message_rank_down(msgId).then((res) => {

                const message = this.discussionArr[this.discussionArr.findIndex(item => item.id == msgId)]
                message.rank = res.new_rank
            }).catch(() => {
                this.$store.state.toast.showToast("Could not rank down message", 4, false)

                console.log("Error: Could not rank down message")
            })

        },
        async updateMessage(msgId, msg, audio_url) {
            await this.edit_message(msgId, msg, audio_url).then(() => {

                const message = this.discussionArr[this.discussionArr.findIndex(item => item.id == msgId)]
                message.content = msg

            }).catch(() => {
                this.$store.state.toast.showToast("Could not update message", 4, false)

                console.log("Error: Could not update message")
            })

        },
        resendMessage(msgId, msg, msg_type) {
            nextTick(() => {
                feather.replace()

            })
            // Resend message
            this.isGenerating = true;
            this.setDiscussionLoading(this.currentDiscussion.id, this.isGenerating);
            axios.get('/get_generation_status', {}).then((res) => {
                if (res) {
                    if (!res.data.status) {
                        socket.emit('generate_msg_from', { prompt: msg, id: msgId, msg_type: msg_type });
                    }
                    else {
                        this.$store.state.toast.showToast("The server is busy. Wait", 4, false)
                        console.log("Already generating");
                    }
                }
            }).catch((error) => {
                console.log("Error: Could not get generation status", error);
            });
        },
        continueMessage(msgId, msg) {
            nextTick(() => {
                feather.replace()

            })
            // Resend message
            this.isGenerating = true;
            this.setDiscussionLoading(this.currentDiscussion.id, this.isGenerating);
            axios.get('/get_generation_status', {}).then((res) => {
                if (res) {
                    if (!res.data.status) {
                        console.log("Continuing generation")
                        socket.emit('continue_generate_msg_from', { prompt: msg, id: msgId });
                    }
                    else {
                        console.log("Already generating");
                    }
                }
            }).catch((error) => {
                console.log("Error: Could not get generation status", error);
            });
        },
        stopGenerating() {
            this.stop_gen()
            this.isGenerating = false
            this.setDiscussionLoading(this.currentDiscussion.id, this.isGenerating)
            nextTick(() => {
                const msgList = document.getElementById('messages-list')
                this.scrollBottom(msgList)
            })
        },
        finalMsgEvent(msgObj) {
            let index=0;
            // Last message contains halucination suppression so we need to update the message content too
            this.discussion_id = msgObj.discussion_id
            if (this.currentDiscussion.id == this.discussion_id) {
                index = this.discussionArr.findIndex((x) => x.id == msgObj.id)
                this.discussionArr[index].content = msgObj.content
                this.discussionArr[index].finished_generating_at = msgObj.finished_generating_at
                this.discussionArr[index].nb_tokens  = msgObj.nb_tokens
                this.discussionArr[index].binding  = msgObj.binding
                this.discussionArr[index].model  = msgObj.model
                this.discussionArr[index].personality  = msgObj.personality


                // const messageItem = this.discussionArr[index]
                // if (messageItem) {
                //     messageItem.content = msgObj.content
                // }
            }
            nextTick(() => {
                const msgList = document.getElementById('messages-list')
                this.scrollBottom(msgList)
                this.recoverFiles()
            })


            this.isGenerating = false
            this.setDiscussionLoading(this.currentDiscussion.id, this.isGenerating)
            this.chime.play()
            index = this.discussionArr.findIndex((x) => x.id == msgObj.id)
            const messageItem = this.discussionArr[index]            
            messageItem.status_message = "Done"
            if(this.$store.state.config.auto_speak && (this.$store.state.config.xtts_enable && this.$store.state.config.xtts_use_streaming_mode)){
                index = this.discussionArr.findIndex((x) => x.id == msgObj.id)
                let message_component = this.$refs['msg-' + msgObj.id][0]
                console.log(message_component)
                message_component.speak()
            }

        },
        copyToClipBoard(messageEntry) {
            let content = ""
            if (messageEntry.message.content) {
                content = messageEntry.message.content
            }


            if(this.$store.state.config.copy_to_clipboard_add_all_details){
                let binding = ""
                if (messageEntry.message.binding) {
                    binding = `Binding: ${messageEntry.message.binding}`
                }
                let personality = ""
                if (messageEntry.message.personality) {
                    personality = `\nPersonality: ${messageEntry.message.personality}`
                }
                let time = ""
                if (messageEntry.created_at_parsed) {
                    time = `\nCreated: ${messageEntry.created_at_parsed}`
                }
                let model = ""
                if (messageEntry.message.model) {
                    model = `Model: ${messageEntry.message.model}`
                }
                let seed = ""
                if (messageEntry.message.seed) {
                    seed = `Seed: ${messageEntry.message.seed}`
                }
                let time_spent = ""
                if (messageEntry.time_spent) {
                    time_spent = `\nTime spent: ${messageEntry.time_spent}`
                }                
                let bottomRow = ''
                bottomRow = `${binding} ${model} ${seed} ${time_spent}`.trim()
                const result = `${messageEntry.message.sender}${personality}${time}\n\n${content}\n\n${bottomRow}`

                navigator.clipboard.writeText(result);
            }
            else{
                navigator.clipboard.writeText(content);
            }



            this.$store.state.toast.showToast("Copied to clipboard successfully", 4, true)

            nextTick(() => {
                feather.replace()

            })
        },
        closeToast() {
            this.showToast = false
        },
        saveJSONtoFile(jsonData, filename) {
            filename = filename || "data.json"
            const a = document.createElement("a");
            a.href = URL.createObjectURL(new Blob([JSON.stringify(jsonData, null, 2)], {
                type: "text/plain"
            }));
            a.setAttribute("download", filename);
            document.body.appendChild(a);
            a.click();
            document.body.removeChild(a);
        },
        saveMarkdowntoFile(markdownData, filename) {
            filename = filename || "data.md"
            const a = document.createElement("a");
            a.href = URL.createObjectURL(new Blob([markdownData], {
                type: "text/plain"
            }));
            a.setAttribute("download", filename);
            document.body.appendChild(a);
            a.click();
            document.body.removeChild(a);
        },
        parseJsonObj(obj) {
            try {
                const ret = JSON.parse(obj)
                return ret
            } catch (error) {
                this.$store.state.toast.showToast("Could not parse JSON. \n" + error.message, 4, false)
                return null
            }

        },
        async parseJsonFile(file) {

            return new Promise((resolve, reject) => {
                const fileReader = new FileReader()
                fileReader.onload = event => resolve(this.parseJsonObj(event.target.result))
                fileReader.onerror = error => reject(error)
                fileReader.readAsText(file)
            })
        },
        
        async exportDiscussionsAsMarkdown() {
            // Export selected discussions

            const discussionIdArr = this.discussionsList.filter((item) => item.checkBoxValue == true).map((item) => {
                return item.id
            })

            if (discussionIdArr.length > 0) {
                let dateObj = new Date()

                const year = dateObj.getFullYear();
                const month = (dateObj.getMonth() + 1).toString().padStart(2, "0");
                const day = dateObj.getDate().toString().padStart(2, "0");
                const hours = dateObj.getHours().toString().padStart(2, "0");
                const minutes = dateObj.getMinutes().toString().padStart(2, "0");
                const seconds = dateObj.getSeconds().toString().padStart(2, "0");
                const formattedDate =
                    year +
                    "." +
                    month +
                    "." +
                    day +
                    "." +
                    hours +
                    "" +
                    minutes +
                    "" +
                    seconds;

                const filename = 'discussions_export_' + formattedDate + '.md'
                this.loading = true
                const res = await this.export_multiple_discussions(discussionIdArr,"markdown")

                if (res) {
                    this.saveMarkdowntoFile(res, filename)
                    this.$store.state.toast.showToast("Successfully exported", 4, true)
                    this.isCheckbox = false
                } else {
                    this.$store.state.toast.showToast("Failed to export discussions", 4, false)
                }
                this.loading = false
            }

        },
        async exportDiscussions(){

        },
        async exportDiscussionsAsJson() {
            // Export selected discussions

            const discussionIdArr = this.discussionsList.filter((item) => item.checkBoxValue == true).map((item) => {
                return item.id
            })

            if (discussionIdArr.length > 0) {
                let dateObj = new Date()

                const year = dateObj.getFullYear();
                const month = (dateObj.getMonth() + 1).toString().padStart(2, "0");
                const day = dateObj.getDate().toString().padStart(2, "0");
                const hours = dateObj.getHours().toString().padStart(2, "0");
                const minutes = dateObj.getMinutes().toString().padStart(2, "0");
                const seconds = dateObj.getSeconds().toString().padStart(2, "0");
                const formattedDate =
                    year +
                    "." +
                    month +
                    "." +
                    day +
                    "." +
                    hours +
                    "" +
                    minutes +
                    "" +
                    seconds;

                const filename = 'discussions_export_' + formattedDate + '.json'
                this.loading = true
                const res = await this.export_multiple_discussions(discussionIdArr, "json")

                if (res) {
                    this.saveJSONtoFile(res, filename)
                    this.$store.state.toast.showToast("Successfully exported", 4, true)
                    this.isCheckbox = false
                } else {
                    this.$store.state.toast.showToast("Failed to export discussions", 4, false)
                }
                this.loading = false
            }

        },
        async importDiscussionsBundle(event){

        },
        async importDiscussions(event) {
            const obj = await this.parseJsonFile(event.target.files[0])

            const res = await this.import_multiple_discussions(obj)
            if (res) {
                this.$store.state.toast.showToast("Successfully imported (" + obj.length + ")", 4, true)
                await this.list_discussions()
            } else {
                this.$store.state.toast.showToast("Failed to import discussions", 4, false)
            }



        },
        async getPersonalityAvatars() {
            while (this.$store.state.personalities === null) {
                await new Promise((resolve) => setTimeout(resolve, 100)); // Wait for 100ms
            }  
            let personalities = this.$store.state.personalities

            this.personalityAvatars = personalities.map(item => {
                const newItem = {
                    name: item.name,
                    avatar: item.avatar
                }
                return newItem
            })
        },
        getAvatar(sender) {
            if (sender.toLowerCase().trim() == this.$store.state.config.user_name.toLowerCase().trim()){
                return "user_infos/"+this.$store.state.config.user_avatar
            }
            const index = this.personalityAvatars.findIndex((x) => x.name === sender)
            const pers = this.personalityAvatars[index]
            if (pers) {
                return pers.avatar
            }

            return
        },
        setFileListChat(files) {


            try {
                this.$refs.chatBox.fileList = this.$refs.chatBox.fileList.concat(files)
            } catch (error) {
                this.$store.state.toast.showToast("Failed to set filelist in chatbox\n" + error.message, 4, false)

            }


            this.isDragOverChat = false


        },
        async setFileListDiscussion(files) {

            if (files.length > 1) {
                this.$store.state.toast.showToast("Failed to import discussions. Too many files", 4, false)
                return
            }
            const obj = await this.parseJsonFile(files[0])

            const res = await this.import_multiple_discussions(obj)
            if (res) {
                this.$store.state.toast.showToast("Successfully imported (" + obj.length + ")", 4, true)
                await this.list_discussions()
            } else {
                this.$store.state.toast.showToast("Failed to import discussions", 4, false)
            }


            this.isDragOverDiscussion = false
        },

    },
    async created() {
        this.randomFact = this.interestingFacts[Math.floor(Math.random() * this.interestingFacts.length)];
        const response = await axios.get('/get_versionID');
        const serverVersionId = response.data.versionId;

        socket.onopen = () => {
            if (this.currentDiscussion!=null){
                this.setPageTitle(this.currentDiscussion.title)
                localStorage.setItem('selected_discussion', this.currentDiscussion.id)
                this.load_discussion(this.currentDiscussion.id, ()=>{
                    if (this.discussionArr.length > 1) {
                        if (this.currentDiscussion.title === '' || this.currentDiscussion.title === null) {
                            this.changeTitleUsingUserMSG(this.currentDiscussion.id, this.discussionArr[1].content)
                        }
                    }
                });
            }
        };

        if (this.versionId !== serverVersionId) {
            // Update the store value
            this.$store.commit('updateVersionId', serverVersionId);
            
            // Force a page refresh
            window.location.reload(true);
        }
        this.$nextTick(() => {
            feather.replace();
        });           
        
        try{
        this.$store.state.loading_infos = "Getting version"
        this.$store.state.loading_progress = 30
        await this.$store.dispatch('getVersion');
        }
        catch (ex){
            console.log("Error cought:", ex)
        }


        try{
        this.$store.state.loading_infos = "Loading Configuration"
        while(socket.id===undefined){
            await new Promise((resolve) => setTimeout(resolve, 100)); // Wait for 100ms
        }
        this.$store.state.client_id = socket.id
        await this.$store.dispatch('refreshConfig');
        }
        catch (ex){
        console.log("Error cought:", ex)
        }
        try{
        this.$store.state.loading_infos = "Loading Database"
        this.$store.state.loading_progress = 20
        await this.$store.dispatch('refreshDatabase');
        }
        catch (ex){
        console.log("Error cought:", ex)
        }

        try{
        this.$store.state.loading_infos = "Getting Bindings list"
        this.$store.state.loading_progress = 40
        await this.$store.dispatch('refreshBindings');
        }
        catch (ex){
        console.log("Error cought:", ex)
        }
    
        try{
        this.$store.state.loading_infos = "Getting personalities zoo"
        this.$store.state.loading_progress = 70
        await this.$store.dispatch('refreshPersonalitiesZoo')
        }
        catch (ex){
        console.log("Error cought:", ex)
        }
        try{
        this.$store.state.loading_infos = "Getting mounted personalities"
        this.$store.state.loading_progress = 80
        await this.$store.dispatch('refreshMountedPersonalities');
        }
        catch (ex){
        console.log("Error cought:", ex)
        }

        try{
        this.$store.state.loading_infos = "Getting models zoo"
        this.$store.state.loading_progress = 90
        await this.$store.dispatch('refreshModelsZoo');
        }
        catch (ex){
        console.log("Error cought:", ex)
        }
        try{
        this.$store.state.loading_infos = "Getting active models"
        this.$store.state.loading_progress = 100
        await this.$store.dispatch('refreshModels');
        await this.$store.dispatch('refreshModelStatus');
        }
        catch (ex){
        console.log("Error cought:", ex)
        }
    
        try{
        await this.$store.dispatch('fetchLanguages');
        await this.$store.dispatch('fetchLanguage');
        }
        catch (ex){
        console.log("Error cought:", ex)
        }
        try{
        await this.$store.dispatch('fetchisRTOn');
        }
        catch (ex){
        console.log("Error cought:", ex)
        }

        this.$store.state.isConnected=true;
        this.$store.state.client_id = socket.id
        // Constructor
        this.setPageTitle()
        await this.list_discussions()
        this.loadLastUsedDiscussion()

        this.isCreated = true
        this.$store.state.ready = true;    

        socket.on('connected',this.socketIOConnected)
        socket.on('disconnected',this.socketIODisconnected)

        // socket responses
        socket.on('show_progress', this.show_progress)
        socket.on('hide_progress', this.hide_progress)
        socket.on('update_progress', this.update_progress)
        
        socket.on('notification', this.notify)
        socket.on('new_message', this.new_message)
        socket.on('update_message', this.update_message)
        socket.on('close_message', this.finalMsgEvent)

        socket.on('disucssion_renamed',(event)=>{
            const index = this.discussionsList.findIndex((x) => x.id == event.discussion_id)
            const discussionItem = this.discussionsList[index]
            discussionItem.title = event.title
            /*
            {
            'status': True,
            'discussion_id':d.id,
            'title':title
            }*/
        })
        socket.onclose = (event) => {
            this.socketIODisconnected();
        };
        socket.on("connect_error", (error) => {
            if (error.message === "ERR_CONNECTION_REFUSED") {
            console.error("Connection refused. The server is not available.");
            // Handle the ERR_CONNECTION_REFUSED error here
            } else {
            console.error("Connection error:", error);
            // Handle other connection errors here
            }
            this.$store.state.isConnected = false
        });        
        socket.onerror = (event) => {
            console.log('WebSocket connection error:', event.code, event.reason);
            this.socketIODisconnected();
            socket.disconnect();
        };
    },
    
    beforeUnmount() {
        window.removeEventListener('resize', this.adjustMenuPosition);
    },
    async mounted() {
        window.addEventListener('keydown', this.handleShortcut);
        this.$store.state.news = this.$refs.news
        this.$store.state.personality_editor = this.$refs.personality_editor

        window.addEventListener('resize', this.adjustMenuPosition);
        
        socket.on('refresh_files',()=>{
            this.recoverFiles()
        })

    },
    async activated() {
        while (this.isReady === false) {
                await new Promise((resolve) => setTimeout(resolve, 100)); // Wait for 100ms
            }  
        await this.getPersonalityAvatars()
        console.log("Avatars found:",this.personalityAvatars)
        if (this.isCreated) {
           // this.loadLastUsedDiscussion()
            nextTick(() => {

                const msgList = document.getElementById('messages-list')

                this.scrollBottom(msgList)

            })
        }
        if (this.$store.state.config.show_news_panel)
            this.$store.state.news.show()
    },
    components: {
        Discussion,
        Message,
        ChatBox,
        WelcomeComponent,
        ChoiceDialog,
        ProgressBar,
        InputBox,
        SkillsLibraryViewer,
        
        PersonalityEditor,
        PopupViewer,
        ActionButton,
        SocialIcon,
        MountedPersonalities,

        ChatBarButton,
        ChangelogPopup

    },
    watch: {        
        installedModels: {
            immediate: true,
            handler(newVal) {
                this.$nextTick(() => {
                this.installedModels = newVal;
                });
            },
        },   
        '$store.state.config.fun_mode': function(newVal, oldVal) {
        },        
        '$store.state.isConnected': function(newVal, oldVal) {
            if (!this.isConnected){
                this.$store.state.messageBox.showBlockingMessage("Server suddenly disconnected. Please reboot the server to recover the connection")
                this.is_first_connection = false
                if(this.$store.state.config.activate_audio_infos)
                    this.connection_lost_audio.play()
            }
            else{
                if(!this.is_first_connection){
                    this.$store.state.messageBox.hideMessage()
                    this.$store.state.messageBox.showMessage("Server connected.")
                    if(this.$store.state.config.activate_audio_infos)
                        this.connection_recovered_audio.play()
                }
            }
            nextTick(() => {
                feather.replace()
            })
        },

        messages: {
        handler: 'extractHtml',
        deep: true
        },
        progress_visibility_val(newVal) {
            console.log("progress_visibility changed to "+ newVal)
        },
        filterTitle(newVal) {
            if (newVal == '') {
                this.filterInProgress = true
                this.discussionsList = this.tempList
                this.filterInProgress = false
            }
        },
        isCheckbox(newval) {
            nextTick(() => {
                feather.replace()
            })
            if (!newval) {
                this.isSelectAll = false
            }
        },
        socketConnected(newval) {
            console.log("Websocket connected (watch)", newval)
        },
        showConfirmation() {
            nextTick(() => {
                feather.replace()

            })
        },
        
    },
    computed: { 
        // Get unique placeholders while preserving order
        parsedPlaceholders() {
            const uniqueMap = new Map();
            this.placeholders.forEach(p => {
                const parsed = parsePlaceholder(p);
                uniqueMap.set(parsed.fullText, parsed);
            });
            return Array.from(uniqueMap.values());
        },       
        filteredBindings() {
            return this.installedBindings.filter(binding => 
            binding.name.toLowerCase().includes(this.bindingSearchQuery.toLowerCase())
            )
        },        
        filteredModels() {
            return this.installedModels.filter(model => 
            model.name.toLowerCase().includes(this.modelSearchQuery.toLowerCase())
            )
        },
        filteredPersonalities() {
            return this.mountedPersonalities.filter(personality => 
            personality.name.toLowerCase().includes(this.personalitySearchQuery.toLowerCase())
            )
        },
        currentModel() {
            return this.$store.state.currentModel || {};
        },
        currentModelIcon() {
            return this.currentModel.icon || this.modelImgPlaceholder;
        },
        binding_name(){
            return this.$store.state.config.binding_name    
        },
        installedModels() {
            return this.$store.state.installedModels;
        },
        model_name(){
            return this.$store.state.config.model_name    
        },
        mountedPersonalities() {
            return this.$store.state.mountedPersArr;
        },
        personality_name(){
            return this.$store.state.config.active_personality_id
        },
        config() {
            return this.$store.state.config;
        },
        mountedPers(){
            return this.$store.state.mountedPers;
        },      
        installedBindings() {
            return this.$store.state.installedBindings;
        },
        currentBindingIcon(){
            return this.currentBinding.icon || this.modelImgPlaceholder;
        },
        currentBinding(){
            return this.$store.state.currentBinding || {};
        },        
        isFullMode() {
          return this.$store.state.view_mode === 'full'; // Accessing the mode directly
        },      
        storeLogo(){
            if (this.$store.state.config){
                return storeLogo
            }
            return this.$store.state.config.app_custom_logo!=''?'/user_infos/'+this.$store.state.config.app_custom_logo:storeLogo
        },        
        currentPersonConfig (){
            try{
                return this.$store.state.currentPersonConfig
            }
            catch{
                console.log("Error finding current personality configuration")
                return undefined
            }
        },        
        selectedPersonality (){
            try{
                return this.$store.state.selectedPersonality
            }
            catch{
                console.log("Error finding current personality configuration")
                return undefined
            }
        },        
        
        loading_infos(){
            return this.$store.state.loading_infos;
        },
        is_fun_mode(){
            try{
                if (this.$store.state.config){
                    return this.$store.state.config.fun_mode;
                }
                else{
                    return false;
                }
            }
            catch(error){
                console.error("Oopsie! Looks like we hit a snag: ", error);
                return false;
            }
        },
        isModelOK(){
            return this.$store.state.isModelOk;
        },
        isGenerating(){
            return this.$store.state.isGenerating;
        },
        isConnected(){
            return this.$store.state.isConnected;
        },        
        ...mapState({
            versionId: state => state.versionId,
        }),
        progress_visibility: {
            get(){
                return self.progress_visibility_val;
            }
        },        
        version_info:{
            get(){
                if(this.$store.state.version!=undefined && this.$store.state.version!="unknown"){
                    return this.$store.state.version;
                }
                else{
                    return "...";
                }
            }
        },
        
        loading_infos:{
            get(){
                return this.$store.state.loading_infos;
            }
        },
        loading_progress:{
            get(){
                return this.$store.state.loading_progress;
            }
        },
        isModelOk:{
            get(){
                return this.$store.state.isModelOk;
            },
            set(val){
                this.$store.state.isModelOk=val
            }
        },
        isGenerating:{
            get(){
                return this.$store.state.isGenerating;
            },
            set(val){
                this.$store.state.isGenerating=val
            }
        },
        personality(){
            const current_personality_name = this.$store.state.config.personalities[this.$store.state.config.active_personality_id].split(":")[0]
            const personality = this.$store.state.personalities.find(personality => personality.full_path === current_personality_name);
            return personality
        },
        prompts_list() {
            return this.personality.prompts_list;
        },
        formatted_database_name() {
            const db_name = this.$store.state.config.discussion_db_name;
            const trimmed_name = db_name;
            return trimmed_name;
        },
        UseDiscussionHistory() {
            return this.$store.state.config.activate_skills_lib;
        }, 
        isReady() {
                return this.$store.state.ready;
        },
        databases(){            
            return this.$store.state.databases;
        },
        client_id() {
            return socket.id
        },
        showLeftPanel() {
           console.log("showLeftPanel")
           console.log(this.$store.state.leftPanelCollapsed)
           return this.$store.state.ready && !this.$store.state.leftPanelCollapsed;
        },
        showRightPanel() {
           console.log("showRightPanel")
           console.log(this.$store.state.rightPanelCollapsed)
           return this.$store.state.ready && !this.$store.state.rightPanelCollapsed;
        },        
        socketConnected() {
            console.log(" --- > Websocket connected")
            this.$store.commit('setIsConnected', true);
            return true
        },
        socketDisconnected() {
            this.$store.commit('setIsConnected', false);
            console.log(" --- > Websocket disconnected")
            return true
        },
        selectedDiscussions() {
            nextTick(() => {
                feather.replace()

            })
            return this.discussionsList.filter((item) => item.checkBoxValue == true)
        }
    }
}
</script>

<script setup>


import Discussion from '../components/Discussion.vue'
import ChoiceDialog from '@/components/ChoiceDialog.vue'
import ProgressBar from "@/components/ProgressBar.vue";
import InputBox from "@/components/input_box.vue";
import SkillsLibraryViewer from "@/components/SkillsViewer.vue"

import Message from '../components/Message.vue'
import ChatBox from '../components/ChatBox.vue'
import WelcomeComponent from '../components/WelcomeComponent.vue'

import socket from '@/services/websocket.js'


import { onMounted } from 'vue'
import { initFlowbite } from 'flowbite'



// initialize components based on data attribute selectors
onMounted(() => {
    initFlowbite()
})

axios.defaults.baseURL = import.meta.env.VITE_LOLLMS_API_BASEURL
</script>
